2017-06-22 01:02:05 +02:00
|
|
|
package bsteam
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"github.com/42wim/matterbridge/bridge/config"
|
|
|
|
"github.com/Philipp15b/go-steam"
|
|
|
|
"github.com/Philipp15b/go-steam/protocol/steamlang"
|
|
|
|
"github.com/Philipp15b/go-steam/steamid"
|
|
|
|
log "github.com/Sirupsen/logrus"
|
|
|
|
//"io/ioutil"
|
|
|
|
"strconv"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Bsteam struct {
|
|
|
|
c *steam.Client
|
|
|
|
connected chan struct{}
|
|
|
|
userMap map[steamid.SteamId]string
|
|
|
|
sync.RWMutex
|
2017-12-19 23:15:03 +01:00
|
|
|
*config.BridgeConfig
|
2017-06-22 01:02:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var flog *log.Entry
|
|
|
|
var protocol = "steam"
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
flog = log.WithFields(log.Fields{"module": protocol})
|
|
|
|
}
|
|
|
|
|
2017-12-19 23:15:03 +01:00
|
|
|
func New(cfg *config.BridgeConfig) *Bsteam {
|
|
|
|
b := &Bsteam{BridgeConfig: cfg}
|
2017-06-22 01:02:05 +02:00
|
|
|
b.userMap = make(map[steamid.SteamId]string)
|
|
|
|
b.connected = make(chan struct{})
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bsteam) Connect() error {
|
|
|
|
flog.Info("Connecting")
|
|
|
|
b.c = steam.NewClient()
|
|
|
|
go b.handleEvents()
|
|
|
|
go b.c.Connect()
|
|
|
|
select {
|
|
|
|
case <-b.connected:
|
|
|
|
flog.Info("Connection succeeded")
|
|
|
|
case <-time.After(time.Second * 30):
|
|
|
|
return fmt.Errorf("connection timed out")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bsteam) Disconnect() error {
|
|
|
|
b.c.Disconnect()
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-08-12 14:51:41 +02:00
|
|
|
func (b *Bsteam) JoinChannel(channel config.ChannelInfo) error {
|
|
|
|
id, err := steamid.NewId(channel.Name)
|
2017-06-22 01:02:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.c.Social.JoinChat(id)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-08-27 22:59:37 +02:00
|
|
|
func (b *Bsteam) Send(msg config.Message) (string, error) {
|
2017-09-11 22:45:15 +02:00
|
|
|
// ignore delete messages
|
|
|
|
if msg.Event == config.EVENT_MSG_DELETE {
|
|
|
|
return "", nil
|
|
|
|
}
|
2017-06-22 01:02:05 +02:00
|
|
|
id, err := steamid.NewId(msg.Channel)
|
|
|
|
if err != nil {
|
2017-08-27 22:59:37 +02:00
|
|
|
return "", err
|
2017-06-22 01:02:05 +02:00
|
|
|
}
|
|
|
|
b.c.Social.SendMessage(id, steamlang.EChatEntryType_ChatMsg, msg.Username+msg.Text)
|
2017-08-27 22:59:37 +02:00
|
|
|
return "", nil
|
2017-06-22 01:02:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bsteam) getNick(id steamid.SteamId) string {
|
|
|
|
b.RLock()
|
|
|
|
defer b.RUnlock()
|
|
|
|
if name, ok := b.userMap[id]; ok {
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
return "unknown"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bsteam) handleEvents() {
|
|
|
|
myLoginInfo := new(steam.LogOnDetails)
|
|
|
|
myLoginInfo.Username = b.Config.Login
|
|
|
|
myLoginInfo.Password = b.Config.Password
|
|
|
|
myLoginInfo.AuthCode = b.Config.AuthCode
|
|
|
|
// Attempt to read existing auth hash to avoid steam guard.
|
|
|
|
// Maybe works
|
|
|
|
//myLoginInfo.SentryFileHash, _ = ioutil.ReadFile("sentry")
|
|
|
|
for event := range b.c.Events() {
|
|
|
|
//flog.Info(event)
|
|
|
|
switch e := event.(type) {
|
|
|
|
case *steam.ChatMsgEvent:
|
|
|
|
flog.Debugf("Receiving ChatMsgEvent: %#v", e)
|
|
|
|
flog.Debugf("Sending message from %s on %s to gateway", b.getNick(e.ChatterId), b.Account)
|
2017-11-08 09:36:20 +10:00
|
|
|
var channel int64
|
|
|
|
if e.ChatRoomId == 0 {
|
|
|
|
channel = int64(e.ChatterId)
|
|
|
|
} else {
|
|
|
|
// for some reason we have to remove 0x18000000000000
|
|
|
|
channel = int64(e.ChatRoomId) - 0x18000000000000
|
|
|
|
}
|
2017-07-03 22:10:26 +02:00
|
|
|
msg := config.Message{Username: b.getNick(e.ChatterId), Text: e.Message, Channel: strconv.FormatInt(channel, 10), Account: b.Account, UserID: strconv.FormatInt(int64(e.ChatterId), 10)}
|
2017-06-22 01:02:05 +02:00
|
|
|
b.Remote <- msg
|
|
|
|
case *steam.PersonaStateEvent:
|
|
|
|
flog.Debugf("PersonaStateEvent: %#v\n", e)
|
|
|
|
b.Lock()
|
|
|
|
b.userMap[e.FriendId] = e.Name
|
|
|
|
b.Unlock()
|
|
|
|
case *steam.ConnectedEvent:
|
|
|
|
b.c.Auth.LogOn(myLoginInfo)
|
|
|
|
case *steam.MachineAuthUpdateEvent:
|
|
|
|
/*
|
|
|
|
flog.Info("authupdate", e)
|
|
|
|
flog.Info("hash", e.Hash)
|
|
|
|
ioutil.WriteFile("sentry", e.Hash, 0666)
|
|
|
|
*/
|
|
|
|
case *steam.LogOnFailedEvent:
|
|
|
|
flog.Info("Logon failed", e)
|
|
|
|
switch e.Result {
|
|
|
|
case steamlang.EResult_AccountLogonDeniedNeedTwoFactorCode:
|
|
|
|
{
|
|
|
|
flog.Info("Steam guard isn't letting me in! Enter 2FA code:")
|
|
|
|
var code string
|
|
|
|
fmt.Scanf("%s", &code)
|
|
|
|
myLoginInfo.TwoFactorCode = code
|
|
|
|
}
|
|
|
|
case steamlang.EResult_AccountLogonDenied:
|
|
|
|
{
|
|
|
|
flog.Info("Steam guard isn't letting me in! Enter auth code:")
|
|
|
|
var code string
|
|
|
|
fmt.Scanf("%s", &code)
|
|
|
|
myLoginInfo.AuthCode = code
|
|
|
|
}
|
|
|
|
default:
|
2017-07-14 00:35:01 +02:00
|
|
|
log.Errorf("LogOnFailedEvent: %#v ", e.Result)
|
2017-06-22 01:02:05 +02:00
|
|
|
// TODO: Handle EResult_InvalidLoginAuthCode
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case *steam.LoggedOnEvent:
|
|
|
|
flog.Debugf("LoggedOnEvent: %#v", e)
|
|
|
|
b.connected <- struct{}{}
|
|
|
|
flog.Debugf("setting online")
|
|
|
|
b.c.Social.SetPersonaState(steamlang.EPersonaState_Online)
|
|
|
|
case *steam.DisconnectedEvent:
|
|
|
|
flog.Info("Disconnected")
|
|
|
|
flog.Info("Attempting to reconnect...")
|
|
|
|
b.c.Connect()
|
|
|
|
case steam.FatalErrorEvent:
|
|
|
|
flog.Error(e)
|
|
|
|
case error:
|
|
|
|
flog.Error(e)
|
|
|
|
default:
|
|
|
|
flog.Debugf("unknown event %#v", e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|