167 lines
3.9 KiB
Go
167 lines
3.9 KiB
Go
package libnmdc
|
|
|
|
import (
|
|
"strings"
|
|
)
|
|
|
|
type adcState int
|
|
|
|
const (
|
|
adcStateProtocol adcState = 0
|
|
adcStateIdentify adcState = 1
|
|
adcStateVerify adcState = 2
|
|
adcStateNormal adcState = 3
|
|
adcStateData adcState = 4
|
|
)
|
|
|
|
type AdcProtocol struct {
|
|
hc *HubConnection
|
|
state adcState
|
|
sid string
|
|
supports map[string]struct{}
|
|
}
|
|
|
|
func NewAdcProtocol(hc *HubConnection) Protocol {
|
|
proto := AdcProtocol{
|
|
hc: hc,
|
|
state: adcStateProtocol,
|
|
supports: make(map[string]struct{}),
|
|
}
|
|
|
|
// Start logging in
|
|
hc.SayRaw("HSUP ADBASE ADTIGR\n")
|
|
|
|
return &proto
|
|
}
|
|
|
|
func (this *AdcProtocol) ProcessCommand(msg string) {
|
|
this.hc.processEvent(HubEvent{EventType: EVENT_DEBUG_MESSAGE, Message: msg})
|
|
|
|
if len(msg) == 0 {
|
|
return
|
|
}
|
|
|
|
parts := strings.Split(msg, " ")
|
|
switch parts[0] {
|
|
|
|
case "ISUP":
|
|
if !(this.state == adcStateProtocol || this.state == adcStateNormal) {
|
|
this.malformed(parts)
|
|
return
|
|
}
|
|
for _, supportflag := range parts[1:] {
|
|
if len(supportflag) < 2 {
|
|
this.malformed(parts)
|
|
return
|
|
}
|
|
|
|
if supportflag[0:2] == "AD" {
|
|
this.supports[supportflag[2:]] = struct{}{}
|
|
} else if supportflag[0:2] == "RM" {
|
|
delete(this.supports, supportflag[2:])
|
|
} else {
|
|
this.malformed(parts)
|
|
return
|
|
}
|
|
}
|
|
if this.state == adcStateProtocol {
|
|
this.state = adcStateIdentify
|
|
}
|
|
|
|
case "ISID":
|
|
if this.state != adcStateIdentify {
|
|
this.malformed(parts)
|
|
return
|
|
}
|
|
this.sid = parts[1]
|
|
|
|
case "IINF":
|
|
flags := make(map[string]string)
|
|
for _, flag := range parts[1:] {
|
|
if len(flag) < 2 {
|
|
this.malformed(parts)
|
|
return
|
|
}
|
|
flags[flag[0:2]] = flag[2:]
|
|
}
|
|
|
|
// Possibilities: User MyINFO (Normal state), or, hub telling information about itself (Identify->Normal state)
|
|
if (this.state == adcStateIdentify || this.state == adcStateVerify) && flags["CT"] == "32" {
|
|
// Hub telling information about itself
|
|
this.handleInfo(flags)
|
|
|
|
// Transition to state VERIFY and send our own info
|
|
this.hc.SayRaw("BINF " + this.escape(this.sid) + " GARBAGE") // FIXME send a real info string
|
|
this.state = adcStateVerify
|
|
|
|
} else if this.state == adcStateNormal {
|
|
// OK
|
|
this.handleInfo(flags)
|
|
|
|
} else {
|
|
this.malformed(parts)
|
|
return
|
|
}
|
|
|
|
case "IGPA":
|
|
//
|
|
|
|
default:
|
|
this.malformed(parts)
|
|
}
|
|
}
|
|
|
|
func (this *AdcProtocol) handleInfo(flags map[string]string) {
|
|
if flags["CT"] == "32" {
|
|
|
|
// IINF DEADCH++\sTest\shub VE2.12.1\s(r"[unknown]")\sRelease HI1 NIADCH++ APADCH++ CT32
|
|
// AP: extension 3.24 "Application and version separation in INF"
|
|
// HI:
|
|
|
|
// Hub properties updated
|
|
hubName, ok := flags["DE"]
|
|
if ok {
|
|
this.hc.HubName = this.unescape(hubName)
|
|
this.hc.processEvent(HubEvent{EventType: EVENT_HUBNAME_CHANGED, Nick: this.hc.HubName})
|
|
}
|
|
|
|
} else {
|
|
// User MyINFO
|
|
// TODO
|
|
|
|
}
|
|
}
|
|
|
|
func (this *AdcProtocol) enterNormalState() {
|
|
this.state = adcStateNormal
|
|
this.hc.processEvent(HubEvent{EventType: EVENT_CONNECTION_STATE_CHANGED, StateChange: CONNECTIONSTATE_CONNECTED})
|
|
this.hc.State = CONNECTIONSTATE_CONNECTED
|
|
}
|
|
|
|
func (this *AdcProtocol) malformed(parts []string) {
|
|
this.hc.processEvent(HubEvent{EventType: EVENT_DEBUG_MESSAGE, Message: "Ignoring malformed, unhandled, or out-of-state protocol command '" + parts[0] + "'"})
|
|
}
|
|
|
|
func (this *AdcProtocol) escape(plaintext string) string {
|
|
// The string "\s" escapes space, "\n" newline and "\\" backslash. This version of the protocol reserves all other escapes for future use; any message containing unknown escapes must be discarded.
|
|
v1 := strings.Replace(plaintext, `\`, `\\`, -1)
|
|
v2 := strings.Replace(v1, "\n", `\n`, -1)
|
|
return strings.Replace(v2, " ", `\s`, -1)
|
|
}
|
|
|
|
func (this *AdcProtocol) unescape(encoded string) string {
|
|
v1 := strings.Replace(encoded, `\s`, " ", -1)
|
|
v2 := strings.Replace(v1, `\n`, "\n", -1)
|
|
return strings.Replace(v2, `\\`, `\`, -1)
|
|
}
|
|
|
|
func (this *AdcProtocol) SayPublic(msg string) {
|
|
}
|
|
|
|
func (this *AdcProtocol) SayPrivate(user, message string) {
|
|
}
|
|
|
|
func (this *AdcProtocol) ProtoMessageSeparator() string {
|
|
return "\n"
|
|
}
|