mirror of
https://github.com/42wim/matterbridge.git
synced 2024-11-25 12:12:05 -08:00
Implement soulseek bridge
This commit is contained in:
parent
d16645c952
commit
cbae29b5dd
87
bridge/soulseek/handlers.go
Normal file
87
bridge/soulseek/handlers.go
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
package bsoulseek
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (b *Bsoulseek) handleMessage(msg soulseekMessageResponse) {
|
||||||
|
if msg != nil {
|
||||||
|
b.Log.Debugf("Handling message: %v", msg)
|
||||||
|
}
|
||||||
|
switch msg := msg.(type) {
|
||||||
|
case loginMessageResponseSuccess, loginMessageResponseFailure:
|
||||||
|
b.loginResponse <- msg
|
||||||
|
case joinRoomMessageResponse:
|
||||||
|
b.joinRoomResponse <- msg
|
||||||
|
case kickedMessageResponse:
|
||||||
|
b.fatalErrors <- fmt.Errorf("Logged in somewhere else")
|
||||||
|
case privateMessageReceive:
|
||||||
|
b.handleDM(msg)
|
||||||
|
case sayChatroomMessageReceive:
|
||||||
|
b.handleChatMessage(msg)
|
||||||
|
case userJoinedRoomMessage:
|
||||||
|
b.handleJoinMessage(msg)
|
||||||
|
case userLeftRoomMessage:
|
||||||
|
b.handleLeaveMessage(msg)
|
||||||
|
default:
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) handleChatMessage(msg sayChatroomMessageReceive) {
|
||||||
|
b.Log.Debugf("Handle chat message: %v", msg)
|
||||||
|
if msg.Username == b.Config.GetString("Nick") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bridgeMessage := config.Message{
|
||||||
|
Account: b.Account,
|
||||||
|
Text: msg.Message,
|
||||||
|
Channel: msg.Room,
|
||||||
|
Username: msg.Username,
|
||||||
|
}
|
||||||
|
b.local <- bridgeMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) handleJoinMessage(msg userJoinedRoomMessage) {
|
||||||
|
b.Log.Debugf("Handle join message: %v", msg)
|
||||||
|
if msg.Username == b.Config.GetString("Nick") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bridgeMessage := config.Message{
|
||||||
|
Account: b.Account,
|
||||||
|
Event: config.EventJoinLeave,
|
||||||
|
Text: fmt.Sprintf("%s has joined the room", msg.Username),
|
||||||
|
Channel: msg.Room,
|
||||||
|
Username: "system",
|
||||||
|
}
|
||||||
|
b.local <- bridgeMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) handleLeaveMessage(msg userLeftRoomMessage) {
|
||||||
|
b.Log.Debugf("Handle leave message: %v", msg)
|
||||||
|
if msg.Username == b.Config.GetString("Nick") {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bridgeMessage := config.Message{
|
||||||
|
Account: b.Account,
|
||||||
|
Event: config.EventJoinLeave,
|
||||||
|
Text: fmt.Sprintf("%s has left the room", msg.Username),
|
||||||
|
Channel: msg.Room,
|
||||||
|
Username: "system",
|
||||||
|
}
|
||||||
|
b.local <- bridgeMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) handleDM(msg privateMessageReceive) {
|
||||||
|
b.Log.Debugf("Received private message: %+v", msg)
|
||||||
|
if msg.Username == "server" {
|
||||||
|
b.Log.Infof("Received system message: %s", msg.Message)
|
||||||
|
if strings.HasPrefix(msg.Message, "System Message: You have been banned") {
|
||||||
|
b.Log.Errorf("Banned from server. Message: %s", msg.Message)
|
||||||
|
b.doDisconnect()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
366
bridge/soulseek/messages.go
Normal file
366
bridge/soulseek/messages.go
Normal file
@ -0,0 +1,366 @@
|
|||||||
|
package bsoulseek
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"crypto/md5"
|
||||||
|
"encoding/binary"
|
||||||
|
"encoding/hex"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
type soulseekMessage interface {}
|
||||||
|
type soulseekMessageResponse interface {}
|
||||||
|
|
||||||
|
const (
|
||||||
|
loginMessageCode uint32 = 1
|
||||||
|
sayInChatRoomMessageCode uint32 = 13
|
||||||
|
joinRoomMessageCode uint32 = 14
|
||||||
|
userJoinedRoomMessageCode uint32 = 16
|
||||||
|
userLeftRoomMessageCode uint32 = 17
|
||||||
|
privateMessageCode uint32 = 22
|
||||||
|
kickedMessageCode uint32 = 41
|
||||||
|
changePasswordMessageCode uint32 = 142
|
||||||
|
)
|
||||||
|
|
||||||
|
var ignoreMessageCodes = map[uint32]bool {
|
||||||
|
7: true,
|
||||||
|
64: true,
|
||||||
|
69: true,
|
||||||
|
83: true,
|
||||||
|
84: true,
|
||||||
|
104: true,
|
||||||
|
113: true,
|
||||||
|
114: true,
|
||||||
|
115: true,
|
||||||
|
130: true,
|
||||||
|
133: true,
|
||||||
|
139: true,
|
||||||
|
140: true,
|
||||||
|
145: true,
|
||||||
|
146: true,
|
||||||
|
148: true,
|
||||||
|
160: true,
|
||||||
|
1003: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 1: Login
|
||||||
|
type loginMessage struct {
|
||||||
|
Code uint32
|
||||||
|
Username string
|
||||||
|
Password string
|
||||||
|
Version uint32
|
||||||
|
Hash string
|
||||||
|
MinorVersion uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type loginMessageResponseSuccess struct {
|
||||||
|
Greet string
|
||||||
|
Address uint32
|
||||||
|
Hash string
|
||||||
|
IsSupporter bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type loginMessageResponseFailure struct {
|
||||||
|
Reason string
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 13: Say in chatroom
|
||||||
|
type sayChatroomMessage struct {
|
||||||
|
Code uint32
|
||||||
|
Room string
|
||||||
|
Message string
|
||||||
|
}
|
||||||
|
|
||||||
|
type sayChatroomMessageReceive struct {
|
||||||
|
Room string
|
||||||
|
Username string
|
||||||
|
Message string
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 14: Join room
|
||||||
|
type joinRoomMessage struct {
|
||||||
|
Code uint32
|
||||||
|
Room string
|
||||||
|
Private uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type userStat struct {
|
||||||
|
AvgSpeed uint32
|
||||||
|
UploadNum uint64
|
||||||
|
Files uint32
|
||||||
|
Dirs uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type joinRoomMessageResponse struct {
|
||||||
|
Room string
|
||||||
|
Users []string
|
||||||
|
Statuses []uint32
|
||||||
|
Stats []userStat
|
||||||
|
SlotsFree []uint32
|
||||||
|
Countries []uint32
|
||||||
|
Owner string
|
||||||
|
Operators []string
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 16: User joined room
|
||||||
|
type userJoinedRoomMessage struct {
|
||||||
|
Room string
|
||||||
|
Username string
|
||||||
|
Status uint32
|
||||||
|
AvgSpeed uint32
|
||||||
|
UploadNum uint64
|
||||||
|
Files uint32
|
||||||
|
Dirs uint32
|
||||||
|
SlotsFree uint32
|
||||||
|
CountryCode string
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 16: User left room
|
||||||
|
type userLeftRoomMessage struct {
|
||||||
|
Room string
|
||||||
|
Username string
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 22: Private message (sometimes used by server to tell us errors)
|
||||||
|
type privateMessageReceive struct {
|
||||||
|
ID uint32
|
||||||
|
Timestamp uint32
|
||||||
|
Username string
|
||||||
|
Message string
|
||||||
|
NewMessage bool
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 41: Kicked from server (relogged)
|
||||||
|
type kickedMessageResponse struct {}
|
||||||
|
|
||||||
|
|
||||||
|
// 142: Change password
|
||||||
|
type changePasswordMessage struct {
|
||||||
|
Code uint32
|
||||||
|
Password string
|
||||||
|
}
|
||||||
|
|
||||||
|
type changePasswordMessageResponse struct {
|
||||||
|
Password string
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func packMessage(message soulseekMessage) ([]byte, error) {
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
var length uint32 = 0
|
||||||
|
binary.Write(buf, binary.LittleEndian, length) // placeholder
|
||||||
|
v := reflect.ValueOf(message)
|
||||||
|
var err error
|
||||||
|
for i := range v.NumField() {
|
||||||
|
val := v.Field(i).Interface()
|
||||||
|
switch val := val.(type) {
|
||||||
|
case string:
|
||||||
|
s_len := uint32(len(val))
|
||||||
|
err = binary.Write(buf, binary.LittleEndian, s_len)
|
||||||
|
buf.WriteString(val)
|
||||||
|
length += s_len + 4
|
||||||
|
case bool, uint8:
|
||||||
|
length += 1
|
||||||
|
err = binary.Write(buf, binary.LittleEndian, val)
|
||||||
|
case uint16:
|
||||||
|
length += 2
|
||||||
|
err = binary.Write(buf, binary.LittleEndian, val)
|
||||||
|
case uint32:
|
||||||
|
length += 4
|
||||||
|
err = binary.Write(buf, binary.LittleEndian, val)
|
||||||
|
case uint64:
|
||||||
|
length += 8
|
||||||
|
err = binary.Write(buf, binary.LittleEndian, val)
|
||||||
|
default:
|
||||||
|
panic("Unsupported struct field type")
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
bytes := buf.Bytes()
|
||||||
|
binary.LittleEndian.PutUint32(bytes, length)
|
||||||
|
return bytes, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpackStructField(reader io.Reader, field reflect.Value) error {
|
||||||
|
switch field.Kind() {
|
||||||
|
case reflect.Struct:
|
||||||
|
for i := range field.NumField() {
|
||||||
|
field := field.Field(i)
|
||||||
|
err := unpackStructField(reader, field)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case reflect.Slice:
|
||||||
|
var length uint32
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &length)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
ilen := int(length)
|
||||||
|
newval := reflect.MakeSlice(field.Type(), ilen, ilen)
|
||||||
|
field.Set(newval)
|
||||||
|
for j := range ilen {
|
||||||
|
err := unpackStructField(reader, field.Index(j))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case reflect.String:
|
||||||
|
var length uint32
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &length)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
val := make([]byte, length)
|
||||||
|
_, err = reader.Read(val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
field.SetString(string(val))
|
||||||
|
case reflect.Bool:
|
||||||
|
var val bool
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
field.SetBool(val)
|
||||||
|
case reflect.Uint8:
|
||||||
|
var val uint8
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
field.SetUint(uint64(val))
|
||||||
|
case reflect.Uint16:
|
||||||
|
var val uint16
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
field.SetUint(uint64(val))
|
||||||
|
case reflect.Uint32:
|
||||||
|
var val uint32
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
field.SetUint(uint64(val))
|
||||||
|
case reflect.Uint64:
|
||||||
|
var val uint64
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
field.SetUint(val)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("Unsupported struct field type: %d", field.Kind()))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func unpackMessage[T soulseekMessage](reader io.Reader) (T, error) {
|
||||||
|
var data T
|
||||||
|
v := reflect.ValueOf(&data).Elem()
|
||||||
|
err := unpackStructField(reader, v)
|
||||||
|
if err != nil && err != io.EOF && err != io.ErrUnexpectedEOF {
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
return data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readMessage(reader io.Reader) (soulseekMessage, error) {
|
||||||
|
var length uint32
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &length)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := make([]byte, int(length))
|
||||||
|
_, err = io.ReadAtLeast(reader, buf, len(buf))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
reader = bytes.NewReader(buf)
|
||||||
|
|
||||||
|
var code uint32
|
||||||
|
err = binary.Read(reader, binary.LittleEndian, &code)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
switch code {
|
||||||
|
case loginMessageCode:
|
||||||
|
// login message is special; has two possible responses
|
||||||
|
var success bool
|
||||||
|
err := binary.Read(reader, binary.LittleEndian, &success)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if success {
|
||||||
|
return unpackMessage[loginMessageResponseSuccess](reader)
|
||||||
|
} else {
|
||||||
|
return unpackMessage[loginMessageResponseFailure](reader)
|
||||||
|
}
|
||||||
|
case sayInChatRoomMessageCode:
|
||||||
|
return unpackMessage[sayChatroomMessageReceive](reader)
|
||||||
|
case joinRoomMessageCode:
|
||||||
|
return unpackMessage[joinRoomMessageResponse](reader)
|
||||||
|
case kickedMessageCode:
|
||||||
|
return unpackMessage[kickedMessageResponse](reader)
|
||||||
|
case userJoinedRoomMessageCode:
|
||||||
|
return unpackMessage[userJoinedRoomMessage](reader)
|
||||||
|
case userLeftRoomMessageCode:
|
||||||
|
return unpackMessage[userLeftRoomMessage](reader)
|
||||||
|
case changePasswordMessageCode:
|
||||||
|
return unpackMessage[changePasswordMessageResponse](reader)
|
||||||
|
case privateMessageCode:
|
||||||
|
return unpackMessage[privateMessageReceive](reader)
|
||||||
|
default:
|
||||||
|
_, ignore := ignoreMessageCodes[code]
|
||||||
|
if ignore {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("Unknown message code: %d", code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeLoginMessage(username string, password string) soulseekMessage {
|
||||||
|
hash := md5.Sum([]byte(username + password))
|
||||||
|
msg := loginMessage{
|
||||||
|
loginMessageCode,
|
||||||
|
username,
|
||||||
|
password,
|
||||||
|
160,
|
||||||
|
hex.EncodeToString(hash[:]),
|
||||||
|
1,
|
||||||
|
}
|
||||||
|
return msg
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeJoinRoomMessage(room string) joinRoomMessage {
|
||||||
|
return joinRoomMessage{
|
||||||
|
joinRoomMessageCode,
|
||||||
|
room,
|
||||||
|
0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeSayChatroomMessage(room string, text string) sayChatroomMessage {
|
||||||
|
return sayChatroomMessage{
|
||||||
|
sayInChatRoomMessageCode,
|
||||||
|
room,
|
||||||
|
text,
|
||||||
|
}
|
||||||
|
}
|
231
bridge/soulseek/soulseek.go
Normal file
231
bridge/soulseek/soulseek.go
Normal file
@ -0,0 +1,231 @@
|
|||||||
|
package bsoulseek
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/42wim/matterbridge/bridge"
|
||||||
|
"github.com/42wim/matterbridge/bridge/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Bsoulseek struct {
|
||||||
|
conn net.Conn
|
||||||
|
messagesToSend chan soulseekMessage
|
||||||
|
local chan config.Message
|
||||||
|
loginResponse chan soulseekMessageResponse
|
||||||
|
joinRoomResponse chan joinRoomMessageResponse
|
||||||
|
fatalErrors chan error
|
||||||
|
disconnect chan bool
|
||||||
|
firstConnectResponse chan error
|
||||||
|
|
||||||
|
*bridge.Config
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(cfg *bridge.Config) bridge.Bridger {
|
||||||
|
b := &Bsoulseek{}
|
||||||
|
b.Config = cfg
|
||||||
|
b.messagesToSend = make(chan soulseekMessage, 256)
|
||||||
|
b.local = make(chan config.Message, 256)
|
||||||
|
b.loginResponse = make(chan soulseekMessageResponse)
|
||||||
|
b.joinRoomResponse = make(chan joinRoomMessageResponse)
|
||||||
|
b.fatalErrors = make(chan error)
|
||||||
|
b.disconnect = make(chan bool)
|
||||||
|
b.firstConnectResponse = make(chan error)
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) receiveMessages() {
|
||||||
|
for {
|
||||||
|
msg, err := readMessage(b.conn)
|
||||||
|
if err != nil {
|
||||||
|
b.fatalErrors <- fmt.Errorf("Reading message failed: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.handleMessage(msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func sliceEqual(s []string) bool {
|
||||||
|
// Return true if every element in s is equal to each other
|
||||||
|
if len(s) <= 1 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
for _, x := range(s) {
|
||||||
|
if x != s[0] {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) sendMessages() {
|
||||||
|
lastFourChatMessages := []string {"1", "2", "3", ""}
|
||||||
|
for {
|
||||||
|
message, more := <-b.messagesToSend
|
||||||
|
if !more {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
msg, is_say := message.(sayChatroomMessage)
|
||||||
|
if is_say {
|
||||||
|
// can't send 5 of the same message in a row or we get banned
|
||||||
|
if (sliceEqual(append(lastFourChatMessages, msg.Message))) {
|
||||||
|
b.Log.Warnf("Dropping message: %s", msg.Message)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
data, err := packMessage(message)
|
||||||
|
if err != nil {
|
||||||
|
b.fatalErrors <- fmt.Errorf("Packing message failed: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
_, err = b.conn.Write(data)
|
||||||
|
if err != nil {
|
||||||
|
b.fatalErrors <- fmt.Errorf("Sending message failed: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Debugf("Sent message: %v", message)
|
||||||
|
if is_say {
|
||||||
|
lastFourChatMessages = append(lastFourChatMessages[1:], msg.Message)
|
||||||
|
time.Sleep(3500 * time.Millisecond) // rate limit so less than 20 can be sent per min
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) sendLocalToRemote() {
|
||||||
|
for {
|
||||||
|
message, more := <-b.local
|
||||||
|
if !more {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Remote <- message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) loginLoop() {
|
||||||
|
firstConnect := true
|
||||||
|
for {
|
||||||
|
if !firstConnect {
|
||||||
|
// Cleanup as we are making new sender/receiver routines
|
||||||
|
b.fatalErrors = make(chan error)
|
||||||
|
close(b.messagesToSend)
|
||||||
|
}
|
||||||
|
// Connect to slsk server
|
||||||
|
server := b.GetString("Server")
|
||||||
|
b.Log.Infof("Connecting %s", server)
|
||||||
|
conn, err := net.Dial("tcp", server)
|
||||||
|
b.conn = conn
|
||||||
|
if err != nil {
|
||||||
|
if firstConnect {
|
||||||
|
b.firstConnectResponse <- err
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Init sender and receiver
|
||||||
|
go b.receiveMessages()
|
||||||
|
go b.sendMessages()
|
||||||
|
go b.sendLocalToRemote()
|
||||||
|
|
||||||
|
// Attempt login
|
||||||
|
b.messagesToSend <- makeLoginMessage(b.GetString("Nick"), b.GetString("Password"))
|
||||||
|
var msg soulseekMessageResponse
|
||||||
|
connected := false
|
||||||
|
select {
|
||||||
|
case msg = <-b.loginResponse:
|
||||||
|
switch msg := msg.(type) {
|
||||||
|
case loginMessageResponseSuccess:
|
||||||
|
if firstConnect {
|
||||||
|
b.firstConnectResponse <- nil
|
||||||
|
}
|
||||||
|
connected = true
|
||||||
|
case loginMessageResponseFailure:
|
||||||
|
if firstConnect {
|
||||||
|
b.firstConnectResponse <- fmt.Errorf("Login failed: %s", msg.Reason)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Errorf("Login failed: %s", msg.Reason)
|
||||||
|
default:
|
||||||
|
panic("Unreachable")
|
||||||
|
}
|
||||||
|
case err := <-b.fatalErrors:
|
||||||
|
// error
|
||||||
|
if firstConnect {
|
||||||
|
b.firstConnectResponse <- fmt.Errorf("Login failed: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Errorf("Login failed: %s", err)
|
||||||
|
case <-time.After(30 * time.Second):
|
||||||
|
// timeout
|
||||||
|
if firstConnect {
|
||||||
|
b.firstConnectResponse <- fmt.Errorf("Login failed: timeout")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b.Log.Errorf("Login failed: timeout")
|
||||||
|
}
|
||||||
|
|
||||||
|
if !connected {
|
||||||
|
// If we reach here, we are not logged in and
|
||||||
|
// it is not the first connect, so we should try again
|
||||||
|
b.Log.Info("Retrying in 30s")
|
||||||
|
time.Sleep(30 * time.Second)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now we are connected
|
||||||
|
select {
|
||||||
|
case err = <- b.fatalErrors:
|
||||||
|
b.Log.Errorf("%s", err)
|
||||||
|
// Retry connect
|
||||||
|
continue
|
||||||
|
case <- b.disconnect:
|
||||||
|
// We are done
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *Bsoulseek) Connect() error {
|
||||||
|
go b.loginLoop()
|
||||||
|
err := <-b.firstConnectResponse
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (b *Bsoulseek) JoinChannel(channel config.ChannelInfo) error {
|
||||||
|
b.messagesToSend <- makeJoinRoomMessage(channel.Name)
|
||||||
|
select {
|
||||||
|
case <-b.joinRoomResponse:
|
||||||
|
b.Log.Infof("Joined room: '%s'", channel.Name)
|
||||||
|
return nil
|
||||||
|
case <-time.After(30 * time.Second):
|
||||||
|
return fmt.Errorf("Could not join room '%s': timeout", channel.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (b *Bsoulseek) Send(msg config.Message) (string, error) {
|
||||||
|
// Only process text messages
|
||||||
|
b.Log.Debugf("=> Received local message %v", msg)
|
||||||
|
if msg.Event != "" && msg.Event != config.EventUserAction && msg.Event != config.EventJoinLeave {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b.messagesToSend <- makeSayChatroomMessage(msg.Channel, msg.Username + msg.Text)
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (b *Bsoulseek) doDisconnect() error {
|
||||||
|
b.disconnect <- true
|
||||||
|
close(b.messagesToSend)
|
||||||
|
close(b.joinRoomResponse)
|
||||||
|
close(b.loginResponse)
|
||||||
|
close(b.local)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (b *Bsoulseek) Disconnect() error {
|
||||||
|
b.doDisconnect()
|
||||||
|
return nil
|
||||||
|
}
|
11
gateway/bridgemap/bsoulseek.go
Normal file
11
gateway/bridgemap/bsoulseek.go
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
// +build !nosoulseek
|
||||||
|
|
||||||
|
package bridgemap
|
||||||
|
|
||||||
|
import (
|
||||||
|
bsoulseek "github.com/42wim/matterbridge/bridge/soulseek"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
FullMap["soulseek"] = bsoulseek.New
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user