package client import ( "encoding/json" "fmt" "io" "maps" "net" "github.com/LandaMm/hsp-go/hsp" ) type ClientOptions struct { Headers map[string]string // TODO: in future support multiple types of auth (credentials, key etc.) Auth string BaseURL string } type Client struct { Options *ClientOptions Base *hsp.Adddress } func NewClient(options *ClientOptions) *Client { if options == nil { options = &ClientOptions{} } var base *hsp.Adddress if len(options.BaseURL) > 0 { addr, err := hsp.ParseAddress(options.BaseURL) if err == nil { base = addr } } return &Client{ Options: options, Base: base, } } func (c *Client) BuildHeaders(address *hsp.Adddress, df *hsp.DataFormat) map[string]string { headers := make(map[string]string) if len(c.Options.Headers) > 0 { maps.Copy(headers, c.Options.Headers) } headers[hsp.H_ROUTE] = address.Route headers[hsp.H_DATA_FORMAT] = df.String() if len(c.Options.Auth) > 0 { headers[hsp.H_AUTH] = c.Options.Auth } return headers } func (c *Client) SingleHit(addr *hsp.Adddress, pkt *hsp.Packet) (*hsp.Packet, error) { rawConn, err := net.Dial("tcp", addr.String()) if err != nil { return nil, err } defer rawConn.Close() keys, err := hsp.GenerateKeyPair() if err != nil { return nil, err } n, err := rawConn.Write(keys.Public[:]) if err != nil { return nil, err } if n != 32 { return nil, fmt.Errorf("failed to send 32 bytes of key (%d sent instead)", n) } serverKey := make([]byte, 32) if _, err := io.ReadFull(rawConn, serverKey); err != nil { return nil, err } sharedKey, err := hsp.DeriveSharedKey(keys.Private, [32]byte(serverKey)) if err != nil { return nil, err } conn := hsp.NewConnection(rawConn, keys, sharedKey) if _, err := conn.Write(pkt); err != nil { return nil, err } return conn.Read() } func (c *Client) SendText(address, text string) (*hsp.Response, error) { var addr *hsp.Adddress var err error if c.Base != nil { addr, err = c.Base.Extend(address) } else { addr, err = hsp.ParseAddress(address) } if err != nil { return nil, err } payload := []byte(text) hdrs := c.BuildHeaders(addr, hsp.TextDataFormat()) pkt := hsp.BuildPacket(hdrs, payload) rpkt, err := c.SingleHit(addr, pkt) if err != nil { return nil, err } return hsp.NewPacketResponse(rpkt), nil } func (c *Client) SendJson(address string, data any) (*hsp.Response, error) { var addr *hsp.Adddress var err error if c.Base != nil { addr, err = c.Base.Extend(address) } else { addr, err = hsp.ParseAddress(address) } if err != nil { return nil, err } payload, err := json.Marshal(data) if err != nil { return nil, err } hdrs := c.BuildHeaders(addr, hsp.JsonDataFormat()) pkt := hsp.BuildPacket(hdrs, payload) rpkt, err := c.SingleHit(addr, pkt) if err != nil { return nil, err } return hsp.NewPacketResponse(rpkt), nil } func (c *Client) SendBytes(address string, data []byte) (*hsp.Response, error) { var addr *hsp.Adddress var err error if c.Base != nil { addr, err = c.Base.Extend(address) } else { addr, err = hsp.ParseAddress(address) } if err != nil { return nil, err } hdrs := c.BuildHeaders(addr, hsp.BytesDataFormat()) pkt := hsp.BuildPacket(hdrs, data) rpkt, err := c.SingleHit(addr, pkt) if err != nil { return nil, err } return hsp.NewPacketResponse(rpkt), nil }