import (
"encoding/binary"
"io"
+ "math/big"
"slices"
)
// FieldOptions flags are sent from v1.5+ clients as part of TranAgreed
const (
- refusePM = 0 // User has "Refuse private messages" pref set
- refuseChat = 1 // User has "Refuse private chat" pref set
- autoResponse = 2 // User has "Automatic response" pref set
+ UserOptRefusePM = 0 // User has "Refuse private messages" pref set
+ UserOptRefuseChat = 1 // User has "Refuse private chat" pref set
+ UserOptAutoResponse = 2 // User has "Automatic response" pref set
)
+type UserFlags [2]byte
+
+func (flag *UserFlags) IsSet(i int) bool {
+ flagBitmap := big.NewInt(int64(binary.BigEndian.Uint16(flag[:])))
+ return flagBitmap.Bit(i) == 1
+}
+
+func (flag *UserFlags) Set(i int, newVal uint) {
+ flagBitmap := big.NewInt(int64(binary.BigEndian.Uint16(flag[:])))
+ flagBitmap.SetBit(flagBitmap, i, newVal)
+ binary.BigEndian.PutUint16(flag[:], uint16(flagBitmap.Int64()))
+}
+
type User struct {
- ID []byte // Size 2
+ ID [2]byte
Icon []byte // Size 2
Flags []byte // Size 2
Name string // Variable length user name
+
+ readOffset int // Internal offset to track read progress
}
func (u *User) Read(p []byte) (int, error) {
u.Flags = u.Flags[2:]
}
- out := append(u.ID[:2], u.Icon[:2]...)
- out = append(out, u.Flags[:2]...)
- out = append(out, nameLen...)
- out = append(out, u.Name...)
-
- return copy(p, slices.Concat(
- u.ID,
+ b := slices.Concat(
+ u.ID[:],
u.Icon,
u.Flags,
nameLen,
[]byte(u.Name),
- )), io.EOF
+ )
+
+ if u.readOffset >= len(b) {
+ return 0, io.EOF // All bytes have been read
+ }
+
+ n := copy(p, b)
+ u.readOffset = n
+
+ return n, nil
}
func (u *User) Write(p []byte) (int, error) {
namelen := int(binary.BigEndian.Uint16(p[6:8]))
- u.ID = p[0:2]
+ u.ID = [2]byte(p[0:2])
u.Icon = p[2:4]
u.Flags = p[4:6]
u.Name = string(p[8 : 8+namelen])
return 8 + namelen, nil
}
-// decodeString decodes an obfuscated user string from a client
-// e.g. 98 8a 9a 8c 8b => "guest"
-func decodeString(obfuText []byte) (clearText string) {
- for _, char := range obfuText {
- clearText += string(rune(255 - uint(char)))
- }
- return clearText
-}
-
// encodeString takes []byte s containing cleartext and rotates by 255 into obfuscated cleartext.
// The Hotline protocol uses this format for sending passwords over network.
// Not secure, but hey, it was the 90s!