mirror of
https://github.com/renorris/openfsd
synced 2026-03-22 14:35:36 +08:00
Changes:
- Implement bootstrapping library for managing several concurrent internal services
- Refactor concurrency model for connections/logical clients and their associated I/O
- Refactor server context singleton
- Refactor error handling
- Most errors are now gracefully sent to the FSD client directly encoded as an $ER packet,
enhancing visibility and debugging
- Most errors are now rightfully treated as non-fatal
- Refactor package/dependency graph
- Refactor calling conventions/interfaces for many packages
- Refactor database package
- Refactor post office
Features:
- Add VATSIM-esque HTTP/JSON "data feed"
- Add ephemeral in-memory database option
- Add user management REST API
- Add improved web interface
- Add MySQL support (drop SQLite support)
100 lines
2.4 KiB
Go
100 lines
2.4 KiB
Go
package handler
|
|
|
|
import (
|
|
"github.com/renorris/openfsd/protocol"
|
|
"strings"
|
|
)
|
|
|
|
// Handler represents a function to process an FSD packet
|
|
type Handler func(invoker Invoker, packet string) (Result, error)
|
|
|
|
// New parses the provided packet's identifier and returns a handler function
|
|
func New(packet string) (handler Handler, err error) {
|
|
// Trim packet delimiter and verify length
|
|
if packet = strings.TrimSuffix(packet, protocol.PacketDelimiter); len(packet) < 3 {
|
|
err = protocol.NewGenericFSDError(protocol.SyntaxError, "", "invalid packet: too short")
|
|
return
|
|
}
|
|
|
|
fields := strings.Split(packet, protocol.Delimiter)
|
|
|
|
switch packet[0] {
|
|
case '^':
|
|
handler = fastPilotPositionHandler
|
|
case '@':
|
|
handler = pilotPositionHandler
|
|
case '$':
|
|
pduID := packet[0:3]
|
|
switch pduID {
|
|
case "$CQ":
|
|
handler = clientQueryHandler
|
|
case "$CR":
|
|
handler = clientQueryResponseHandler
|
|
case "$ZC":
|
|
handler = authChallengeHandler
|
|
case "$ZR":
|
|
handler = authChallengeResponseHandler
|
|
case "$!!":
|
|
handler = killRequestHandler
|
|
case "$PI":
|
|
handler = pingHandler
|
|
case "$AX":
|
|
// TODO: implement METAR request
|
|
}
|
|
case '#':
|
|
pduID := packet[0:3]
|
|
switch pduID {
|
|
case "#SL":
|
|
handler = fastPilotPositionHandler
|
|
case "#ST":
|
|
handler = fastPilotPositionHandler
|
|
case "#DP":
|
|
handler = deletePilotHandler
|
|
case "#SB":
|
|
if len(fields) < 3 {
|
|
err = protocol.NewGenericFSDError(protocol.SyntaxError, "", "unrecognized #SB packet: invalid parameter count")
|
|
return
|
|
}
|
|
switch fields[2] {
|
|
case "PIR":
|
|
handler = planeInfoRequestHandler
|
|
case "FSIPIR":
|
|
handler = planeInfoRequestFsinnHandler
|
|
case "PI":
|
|
if len(fields) > 3 && fields[3] == "GEN" {
|
|
handler = planeInfoResponseHandler
|
|
}
|
|
}
|
|
case "#TM":
|
|
if len(fields) < 3 {
|
|
err = protocol.NewGenericFSDError(protocol.SyntaxError, "", "unrecognized #TM packet: invalid parameter count")
|
|
return
|
|
}
|
|
switch fields[1] {
|
|
case "*":
|
|
handler = broadcastMessageHandler
|
|
case "*S":
|
|
handler = wallopMessageHandler
|
|
default:
|
|
if len(fields[1]) > 0 && fields[1][0] == '@' {
|
|
handler = radioMessageHandler
|
|
} else {
|
|
handler = directMessageHandler
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if handler == nil {
|
|
err = protocol.NewGenericFSDError(protocol.SyntaxError, "", "unrecognized packet type")
|
|
return
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func pduSourceInvalidResult() (result Result, err error) {
|
|
err = protocol.NewGenericFSDError(protocol.PDUSourceInvalidError, "", "")
|
|
return
|
|
}
|