commands.go 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. package encoding
  2. import (
  3. "encoding/binary"
  4. "io"
  5. "github.com/xtls/xray-core/common"
  6. "github.com/xtls/xray-core/common/buf"
  7. "github.com/xtls/xray-core/common/errors"
  8. "github.com/xtls/xray-core/common/protocol"
  9. )
  10. var (
  11. ErrCommandTooLarge = errors.New("Command too large.")
  12. ErrCommandTypeMismatch = errors.New("Command type mismatch.")
  13. ErrInvalidAuth = errors.New("Invalid auth.")
  14. ErrInsufficientLength = errors.New("Insufficient length.")
  15. ErrUnknownCommand = errors.New("Unknown command.")
  16. )
  17. func MarshalCommand(command interface{}, writer io.Writer) error {
  18. if command == nil {
  19. return ErrUnknownCommand
  20. }
  21. var cmdID byte
  22. var factory CommandFactory
  23. switch command.(type) {
  24. default:
  25. return ErrUnknownCommand
  26. }
  27. buffer := buf.New()
  28. defer buffer.Release()
  29. err := factory.Marshal(command, buffer)
  30. if err != nil {
  31. return err
  32. }
  33. auth := Authenticate(buffer.Bytes())
  34. length := buffer.Len() + 4
  35. if length > 255 {
  36. return ErrCommandTooLarge
  37. }
  38. common.Must2(writer.Write([]byte{cmdID, byte(length), byte(auth >> 24), byte(auth >> 16), byte(auth >> 8), byte(auth)}))
  39. common.Must2(writer.Write(buffer.Bytes()))
  40. return nil
  41. }
  42. func UnmarshalCommand(cmdID byte, data []byte) (protocol.ResponseCommand, error) {
  43. if len(data) <= 4 {
  44. return nil, ErrInsufficientLength
  45. }
  46. expectedAuth := Authenticate(data[4:])
  47. actualAuth := binary.BigEndian.Uint32(data[:4])
  48. if expectedAuth != actualAuth {
  49. return nil, ErrInvalidAuth
  50. }
  51. var factory CommandFactory
  52. switch cmdID {
  53. default:
  54. return nil, ErrUnknownCommand
  55. }
  56. return factory.Unmarshal(data[4:])
  57. }
  58. type CommandFactory interface {
  59. Marshal(command interface{}, writer io.Writer) error
  60. Unmarshal(data []byte) (interface{}, error)
  61. }