mirror of
https://github.com/42wim/matterbridge.git
synced 2024-11-26 12:42:03 -08:00
366 lines
7.3 KiB
Go
366 lines
7.3 KiB
Go
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,
|
|
}
|
|
} |