From 9990e92f19006017349d40f087ff08239ab53d3e Mon Sep 17 00:00:00 2001 From: Elara Musayelyan Date: Fri, 22 Apr 2022 17:12:30 -0700 Subject: [PATCH] Switch from custom socket API to rpcx --- api/api.go | 117 +++++ api/client.go | 153 ------ api/firmware.go | 40 ++ api/fs.go | 153 +++--- api/get.go | 67 +++ api/info.go | 209 -------- api/notify.go | 17 +- api/set.go | 15 + api/time.go | 33 -- api/types.go | 96 ++++ api/update.go | 12 + api/upgrade.go | 48 -- api/watch.go | 144 ++++++ api/weather.go | 17 - cmd/itctl/firmware.go | 9 +- cmd/itctl/fs.go | 15 +- cmd/itctl/set.go | 2 +- cmd/itctl/update.go | 2 +- cmd/itgui/time.go | 2 +- cmd/itgui/upgrade.go | 7 +- go.mod | 10 +- go.sum | 514 ++++++++++++++++++- internal/types/types.go | 150 ------ socket.go | 1048 +++++++++++++++++---------------------- 24 files changed, 1569 insertions(+), 1311 deletions(-) create mode 100644 api/api.go delete mode 100644 api/client.go create mode 100644 api/firmware.go create mode 100644 api/get.go delete mode 100644 api/info.go create mode 100644 api/set.go delete mode 100644 api/time.go create mode 100644 api/types.go create mode 100644 api/update.go delete mode 100644 api/upgrade.go create mode 100644 api/watch.go delete mode 100644 api/weather.go delete mode 100644 internal/types/types.go diff --git a/api/api.go b/api/api.go new file mode 100644 index 0000000..9492c74 --- /dev/null +++ b/api/api.go @@ -0,0 +1,117 @@ +package api + +import ( + "context" + + "github.com/smallnest/rpcx/client" + "github.com/smallnest/rpcx/protocol" + "github.com/vmihailenco/msgpack/v5" + "go.arsenm.dev/infinitime" +) + +const DefaultAddr = "/tmp/itd/socket" + +type Client struct { + itdClient client.XClient + itdCh chan *protocol.Message + fsClient client.XClient + fsCh chan *protocol.Message + srvVals map[string]chan interface{} +} + +func New(sockPath string) (*Client, error) { + d, err := client.NewPeer2PeerDiscovery("unix@"+sockPath, "") + if err != nil { + return nil, err + } + + out := &Client{} + + out.itdCh = make(chan *protocol.Message, 5) + out.itdClient = client.NewBidirectionalXClient( + "ITD", + client.Failtry, + client.RandomSelect, + d, + client.DefaultOption, + out.itdCh, + ) + + out.fsCh = make(chan *protocol.Message, 5) + out.fsClient = client.NewBidirectionalXClient( + "FS", + client.Failtry, + client.RandomSelect, + d, + client.DefaultOption, + out.fsCh, + ) + + out.srvVals = map[string]chan interface{}{} + + go out.handleMessages(out.itdCh) + go out.handleMessages(out.fsCh) + + return out, nil +} + +func (c *Client) handleMessages(msgCh chan *protocol.Message) { + for msg := range msgCh { + _, ok := c.srvVals[msg.ServicePath] + if !ok { + c.srvVals[msg.ServicePath] = make(chan interface{}, 5) + } + + //fmt.Printf("%+v\n", msg) + + ch := c.srvVals[msg.ServicePath] + + switch msg.ServiceMethod { + case "FSProgress": + var progress FSTransferProgress + msgpack.Unmarshal(msg.Payload, &progress) + ch <- progress + case "DFUProgress": + var progress infinitime.DFUProgress + msgpack.Unmarshal(msg.Payload, &progress) + ch <- progress + case "MotionSample": + var motionVals infinitime.MotionValues + msgpack.Unmarshal(msg.Payload, &motionVals) + ch <- motionVals + case "Done": + close(c.srvVals[msg.ServicePath]) + delete(c.srvVals, msg.ServicePath) + default: + var value interface{} + msgpack.Unmarshal(msg.Payload, &value) + ch <- value + } + } +} + +func (c *Client) done(id string) error { + return c.itdClient.Call( + context.Background(), + "Done", + id, + nil, + ) +} + +func (c *Client) Close() error { + err := c.itdClient.Close() + if err != nil { + return err + } + + err = c.fsClient.Close() + if err != nil { + return err + } + + close(c.itdCh) + close(c.fsCh) + + return nil +} diff --git a/api/client.go b/api/client.go deleted file mode 100644 index 16eaad3..0000000 --- a/api/client.go +++ /dev/null @@ -1,153 +0,0 @@ -package api - -import ( - "bufio" - "encoding/json" - "errors" - "net" - - "github.com/mitchellh/mapstructure" - "go.arsenm.dev/infinitime" - "go.arsenm.dev/itd/internal/types" -) - -// Default socket address -const DefaultAddr = "/tmp/itd/socket" - -// Client is the socket API client -type Client struct { - conn net.Conn - respCh chan types.Response - heartRateCh chan types.Response - battLevelCh chan types.Response - stepCountCh chan types.Response - motionCh chan types.Response - dfuProgressCh chan types.Response - readProgressCh chan types.FSTransferProgress - writeProgressCh chan types.FSTransferProgress -} - -// New creates a new client and sets it up -func New(addr string) (*Client, error) { - conn, err := net.Dial("unix", addr) - if err != nil { - return nil, err - } - - out := &Client{ - conn: conn, - respCh: make(chan types.Response, 5), - } - - go func() { - scanner := bufio.NewScanner(conn) - for scanner.Scan() { - var res types.Response - err = json.Unmarshal(scanner.Bytes(), &res) - if err != nil { - continue - } - out.handleResp(res) - } - }() - return out, err -} - -func (c *Client) Close() error { - err := c.conn.Close() - if err != nil { - return err - } - close(c.respCh) - return nil -} - -// request sends a request to itd and waits for and returns the response -func (c *Client) request(req types.Request) (types.Response, error) { - // Encode request into connection - err := json.NewEncoder(c.conn).Encode(req) - if err != nil { - return types.Response{}, err - } - - res := <-c.respCh - - if res.Error { - return res, errors.New(res.Message) - } - - return res, nil -} - -// requestNoRes sends a request to itd and does not wait for the response -func (c *Client) requestNoRes(req types.Request) error { - // Encode request into connection - err := json.NewEncoder(c.conn).Encode(req) - if err != nil { - return err - } - return nil -} - -// handleResp handles the received response as needed -func (c *Client) handleResp(res types.Response) error { - switch res.Type { - case types.ReqTypeWatchHeartRate: - c.heartRateCh <- res - case types.ReqTypeWatchBattLevel: - c.battLevelCh <- res - case types.ReqTypeWatchStepCount: - c.stepCountCh <- res - case types.ReqTypeWatchMotion: - c.motionCh <- res - case types.ReqTypeFwUpgrade: - c.dfuProgressCh <- res - case types.ReqTypeFS: - if res.Value == nil { - c.respCh <- res - break - } - var progress types.FSTransferProgress - if err := mapstructure.Decode(res.Value, &progress); err != nil { - c.respCh <- res - break - } - switch progress.Type { - case types.FSTypeRead: - c.readProgressCh <- progress - case types.FSTypeWrite: - c.writeProgressCh <- progress - default: - c.respCh <- res - } - default: - c.respCh <- res - } - return nil -} - -func decodeUint8(val interface{}) uint8 { - return uint8(val.(float64)) -} - -func decodeUint32(val interface{}) uint32 { - return uint32(val.(float64)) -} - -func decodeMotion(val interface{}) (infinitime.MotionValues, error) { - out := infinitime.MotionValues{} - err := mapstructure.Decode(val, &out) - if err != nil { - return out, err - } - return out, nil -} - -func decodeDFUProgress(val interface{}) (DFUProgress, error) { - out := DFUProgress{} - err := mapstructure.Decode(val, &out) - if err != nil { - return out, err - } - return out, nil -} diff --git a/api/firmware.go b/api/firmware.go new file mode 100644 index 0000000..1335365 --- /dev/null +++ b/api/firmware.go @@ -0,0 +1,40 @@ +package api + +import ( + "context" + "time" + + "go.arsenm.dev/infinitime" +) + +func (c *Client) FirmwareUpgrade(upgType UpgradeType, files ...string) (chan infinitime.DFUProgress, error) { + var id string + err := c.itdClient.Call( + context.Background(), + "FirmwareUpgrade", + FwUpgradeData{ + Type: upgType, + Files: files, + }, + &id, + ) + if err != nil { + return nil, err + } + + progressCh := make(chan infinitime.DFUProgress, 5) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + progressCh <- val.(infinitime.DFUProgress) + } + close(progressCh) + }() + + return progressCh, nil +} diff --git a/api/fs.go b/api/fs.go index 1102e11..735aeff 100644 --- a/api/fs.go +++ b/api/fs.go @@ -1,102 +1,101 @@ package api import ( - "github.com/mitchellh/mapstructure" - "go.arsenm.dev/itd/internal/types" + "context" + "time" ) -func (c *Client) Rename(old, new string) error { - _, err := c.request(types.Request{ - Type: types.ReqTypeFS, - Data: types.ReqDataFS{ - Type: types.FSTypeMove, - Files: []string{old, new}, - }, - }) - if err != nil { - return err - } - return nil +func (c *Client) Remove(paths ...string) error { + return c.fsClient.Call( + context.Background(), + "Remove", + paths, + nil, + ) } -func (c *Client) Remove(paths ...string) error { - _, err := c.request(types.Request{ - Type: types.ReqTypeFS, - Data: types.ReqDataFS{ - Type: types.FSTypeDelete, - Files: paths, - }, - }) - if err != nil { - return err - } - return nil +func (c *Client) Rename(old, new string) error { + return c.fsClient.Call( + context.Background(), + "Remove", + [2]string{old, new}, + nil, + ) } func (c *Client) Mkdir(paths ...string) error { - _, err := c.request(types.Request{ - Type: types.ReqTypeFS, - Data: types.ReqDataFS{ - Type: types.FSTypeMkdir, - Files: paths, - }, - }) - if err != nil { - return err - } - return nil + return c.fsClient.Call( + context.Background(), + "Mkdir", + paths, + nil, + ) } -func (c *Client) ReadDir(path string) ([]types.FileInfo, error) { - res, err := c.request(types.Request{ - Type: types.ReqTypeFS, - Data: types.ReqDataFS{ - Type: types.FSTypeList, - Files: []string{path}, - }, - }) - if err != nil { - return nil, err - } - var out []types.FileInfo - err = mapstructure.Decode(res.Value, &out) - if err != nil { - return nil, err - } - return out, nil +func (c *Client) ReadDir(dir string) (out []FileInfo, err error) { + err = c.fsClient.Call( + context.Background(), + "ReadDir", + dir, + &out, + ) + return } -func (c *Client) ReadFile(localPath, remotePath string) (<-chan types.FSTransferProgress, error) { - c.readProgressCh = make(chan types.FSTransferProgress, 5) - - _, err := c.request(types.Request{ - Type: types.ReqTypeFS, - Data: types.ReqDataFS{ - Type: types.FSTypeRead, - Files: []string{localPath, remotePath}, - }, - }) - +func (c *Client) Upload(dst, src string) (chan FSTransferProgress, error) { + var id string + err := c.fsClient.Call( + context.Background(), + "Upload", + [2]string{dst, src}, + &id, + ) if err != nil { return nil, err } - return c.readProgressCh, nil + progressCh := make(chan FSTransferProgress, 5) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + progressCh <- val.(FSTransferProgress) + } + close(progressCh) + }() + + return progressCh, nil } -func (c *Client) WriteFile(localPath, remotePath string) (<-chan types.FSTransferProgress, error) { - c.writeProgressCh = make(chan types.FSTransferProgress, 5) - - _, err := c.request(types.Request{ - Type: types.ReqTypeFS, - Data: types.ReqDataFS{ - Type: types.FSTypeWrite, - Files: []string{remotePath, localPath}, - }, - }) +func (c *Client) Download(dst, src string) (chan FSTransferProgress, error) { + var id string + err := c.fsClient.Call( + context.Background(), + "Download", + [2]string{dst, src}, + &id, + ) if err != nil { return nil, err } - return c.writeProgressCh, nil + progressCh := make(chan FSTransferProgress, 5) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + progressCh <- val.(FSTransferProgress) + } + close(progressCh) + }() + + return progressCh, nil } diff --git a/api/get.go b/api/get.go new file mode 100644 index 0000000..9229ef8 --- /dev/null +++ b/api/get.go @@ -0,0 +1,67 @@ +package api + +import ( + "context" + + "go.arsenm.dev/infinitime" +) + +func (c *Client) HeartRate() (out uint8, err error) { + err = c.itdClient.Call( + context.Background(), + "HeartRate", + nil, + &out, + ) + return +} + +func (c *Client) BatteryLevel() (out uint8, err error) { + err = c.itdClient.Call( + context.Background(), + "BatteryLevel", + nil, + &out, + ) + return +} + +func (c *Client) Motion() (out infinitime.MotionValues, err error) { + err = c.itdClient.Call( + context.Background(), + "Motion", + nil, + &out, + ) + return +} + +func (c *Client) StepCount() (out uint32, err error) { + err = c.itdClient.Call( + context.Background(), + "StepCount", + nil, + &out, + ) + return +} + +func (c *Client) Version() (out string, err error) { + err = c.itdClient.Call( + context.Background(), + "Version", + nil, + &out, + ) + return +} + +func (c *Client) Address() (out string, err error) { + err = c.itdClient.Call( + context.Background(), + "Address", + nil, + &out, + ) + return +} diff --git a/api/info.go b/api/info.go deleted file mode 100644 index 98670a2..0000000 --- a/api/info.go +++ /dev/null @@ -1,209 +0,0 @@ -package api - -import ( - "github.com/mitchellh/mapstructure" - "go.arsenm.dev/infinitime" - "go.arsenm.dev/itd/internal/types" -) - -// Address gets the bluetooth address of the connected device -func (c *Client) Address() (string, error) { - res, err := c.request(types.Request{ - Type: types.ReqTypeBtAddress, - }) - if err != nil { - return "", err - } - - return res.Value.(string), nil -} - -// Version gets the firmware version of the connected device -func (c *Client) Version() (string, error) { - res, err := c.request(types.Request{ - Type: types.ReqTypeFwVersion, - }) - if err != nil { - return "", err - } - - return res.Value.(string), nil -} - -// BatteryLevel gets the battery level of the connected device -func (c *Client) BatteryLevel() (uint8, error) { - res, err := c.request(types.Request{ - Type: types.ReqTypeBattLevel, - }) - if err != nil { - return 0, err - } - - return uint8(res.Value.(float64)), nil -} - -// WatchBatteryLevel returns a channel which will contain -// new battery level values as they update. Do not use after -// calling cancellation function -func (c *Client) WatchBatteryLevel() (<-chan uint8, func(), error) { - c.battLevelCh = make(chan types.Response, 2) - err := c.requestNoRes(types.Request{ - Type: types.ReqTypeWatchBattLevel, - }) - if err != nil { - return nil, nil, err - } - res := <-c.battLevelCh - done, cancel := c.cancelFn(res.ID, c.battLevelCh) - out := make(chan uint8, 2) - go func() { - for res := range c.battLevelCh { - select { - case <-done: - return - default: - out <- decodeUint8(res.Value) - } - } - }() - return out, cancel, nil -} - -// HeartRate gets the heart rate from the connected device -func (c *Client) HeartRate() (uint8, error) { - res, err := c.request(types.Request{ - Type: types.ReqTypeHeartRate, - }) - if err != nil { - return 0, err - } - - return decodeUint8(res.Value), nil -} - -// WatchHeartRate returns a channel which will contain -// new heart rate values as they update. Do not use after -// calling cancellation function -func (c *Client) WatchHeartRate() (<-chan uint8, func(), error) { - c.heartRateCh = make(chan types.Response, 2) - err := c.requestNoRes(types.Request{ - Type: types.ReqTypeWatchHeartRate, - }) - if err != nil { - return nil, nil, err - } - res := <-c.heartRateCh - done, cancel := c.cancelFn(res.ID, c.heartRateCh) - out := make(chan uint8, 2) - go func() { - for res := range c.heartRateCh { - select { - case <-done: - return - default: - out <- decodeUint8(res.Value) - } - } - }() - return out, cancel, nil -} - -// cancelFn generates a cancellation function for the given -// request type and channel -func (c *Client) cancelFn(reqID string, ch chan types.Response) (chan struct{}, func()) { - done := make(chan struct{}, 1) - return done, func() { - done <- struct{}{} - close(ch) - c.requestNoRes(types.Request{ - Type: types.ReqTypeCancel, - Data: reqID, - }) - } -} - -// StepCount gets the step count from the connected device -func (c *Client) StepCount() (uint32, error) { - res, err := c.request(types.Request{ - Type: types.ReqTypeStepCount, - }) - if err != nil { - return 0, err - } - - return uint32(res.Value.(float64)), nil -} - -// WatchStepCount returns a channel which will contain -// new step count values as they update. Do not use after -// calling cancellation function -func (c *Client) WatchStepCount() (<-chan uint32, func(), error) { - c.stepCountCh = make(chan types.Response, 2) - err := c.requestNoRes(types.Request{ - Type: types.ReqTypeWatchStepCount, - }) - if err != nil { - return nil, nil, err - } - res := <-c.stepCountCh - done, cancel := c.cancelFn(res.ID, c.stepCountCh) - out := make(chan uint32, 2) - go func() { - for res := range c.stepCountCh { - select { - case <-done: - return - default: - out <- decodeUint32(res.Value) - } - } - }() - return out, cancel, nil -} - -// Motion gets the motion values from the connected device -func (c *Client) Motion() (infinitime.MotionValues, error) { - out := infinitime.MotionValues{} - res, err := c.request(types.Request{ - Type: types.ReqTypeMotion, - }) - if err != nil { - return out, err - } - err = mapstructure.Decode(res.Value, &out) - if err != nil { - return out, err - } - return out, nil -} - -// WatchMotion returns a channel which will contain -// new motion values as they update. Do not use after -// calling cancellation function -func (c *Client) WatchMotion() (<-chan infinitime.MotionValues, func(), error) { - c.motionCh = make(chan types.Response, 5) - err := c.requestNoRes(types.Request{ - Type: types.ReqTypeWatchMotion, - }) - if err != nil { - return nil, nil, err - } - res := <-c.motionCh - done, cancel := c.cancelFn(res.ID, c.motionCh) - out := make(chan infinitime.MotionValues, 5) - go func() { - for res := range c.motionCh { - select { - case <-done: - return - default: - motion, err := decodeMotion(res.Value) - if err != nil { - continue - } - out <- motion - } - } - }() - return out, cancel, nil -} diff --git a/api/notify.go b/api/notify.go index ccc2b6e..ca48c4c 100644 --- a/api/notify.go +++ b/api/notify.go @@ -1,14 +1,17 @@ package api -import "go.arsenm.dev/itd/internal/types" +import ( + "context" +) -func (c *Client) Notify(title string, body string) error { - _, err := c.request(types.Request{ - Type: types.ReqTypeNotify, - Data: types.ReqDataNotify{ +func (c *Client) Notify(title, body string) error { + return c.itdClient.Call( + context.Background(), + "Notify", + NotifyData{ Title: title, Body: body, }, - }) - return err + nil, + ) } \ No newline at end of file diff --git a/api/set.go b/api/set.go new file mode 100644 index 0000000..c6918d6 --- /dev/null +++ b/api/set.go @@ -0,0 +1,15 @@ +package api + +import ( + "context" + "time" +) + +func (c *Client) SetTime(t time.Time) error { + return c.itdClient.Call( + context.Background(), + "SetTime", + t, + nil, + ) +} diff --git a/api/time.go b/api/time.go deleted file mode 100644 index 4661204..0000000 --- a/api/time.go +++ /dev/null @@ -1,33 +0,0 @@ -package api - -import ( - "time" - - "go.arsenm.dev/itd/internal/types" -) - -// SetTime sets the given time on the connected device -func (c *Client) SetTime(t time.Time) error { - _, err := c.request(types.Request{ - Type: types.ReqTypeSetTime, - Data: t.Format(time.RFC3339), - }) - if err != nil { - return err - } - return nil -} - -// SetTimeNow sets the time on the connected device to -// the current time. This is more accurate than -// SetTime(time.Now()) due to RFC3339 formatting -func (c *Client) SetTimeNow() error { - _, err := c.request(types.Request{ - Type: types.ReqTypeSetTime, - Data: "now", - }) - if err != nil { - return err - } - return nil -} diff --git a/api/types.go b/api/types.go new file mode 100644 index 0000000..281a85b --- /dev/null +++ b/api/types.go @@ -0,0 +1,96 @@ +package api + +import ( + "fmt" + "strconv" +) + +type UpgradeType uint8 + +const ( + UpgradeTypeArchive UpgradeType = iota + UpgradeTypeFiles +) + +type FSData struct { + Files []string + Data string +} + +type FwUpgradeData struct { + Type UpgradeType + Files []string +} + +type NotifyData struct { + Title string + Body string +} + +type FSTransferProgress struct { + Total uint32 + Sent uint32 +} + +type FileInfo struct { + Name string + Size int64 + IsDir bool +} + +func (fi FileInfo) String() string { + var isDirChar rune + if fi.IsDir { + isDirChar = 'd' + } else { + isDirChar = '-' + } + + // Get human-readable value for file size + val, unit := bytesHuman(fi.Size) + prec := 0 + // If value is less than 10, set precision to 1 + if val < 10 { + prec = 1 + } + // Convert float to string + valStr := strconv.FormatFloat(val, 'f', prec, 64) + + // Return string formatted like so: + // - 10 kB file + // or: + // d 0 B . + return fmt.Sprintf( + "%c %3s %-2s %s", + isDirChar, + valStr, + unit, + fi.Name, + ) +} + +// bytesHuman returns a human-readable string for +// the amount of bytes inputted. +func bytesHuman(b int64) (float64, string) { + const unit = 1000 + // Set possible units prefixes (PineTime flash is 4MB) + units := [2]rune{'k', 'M'} + // If amount of bytes is less than smallest unit + if b < unit { + // Return unchanged with unit "B" + return float64(b), "B" + } + + div, exp := int64(unit), 0 + // Get decimal values and unit prefix index + for n := b / unit; n >= unit; n /= unit { + div *= unit + exp++ + } + + // Create string for full unit + unitStr := string([]rune{units[exp], 'B'}) + + // Return decimal with unit string + return float64(b) / float64(div), unitStr +} diff --git a/api/update.go b/api/update.go new file mode 100644 index 0000000..5ff299c --- /dev/null +++ b/api/update.go @@ -0,0 +1,12 @@ +package api + +import "context" + +func (c *Client) WeatherUpdate() error { + return c.itdClient.Call( + context.Background(), + "WeatherUpdate", + nil, + nil, + ) +} \ No newline at end of file diff --git a/api/upgrade.go b/api/upgrade.go deleted file mode 100644 index 1f1d44e..0000000 --- a/api/upgrade.go +++ /dev/null @@ -1,48 +0,0 @@ -package api - -import ( - "encoding/json" - - "go.arsenm.dev/itd/internal/types" -) - -// DFUProgress stores the progress of a DFU upfate -type DFUProgress types.DFUProgress - -// UpgradeType indicates the type of upgrade to be performed -type UpgradeType uint8 - -// Type of DFU upgrade -const ( - UpgradeTypeArchive UpgradeType = iota - UpgradeTypeFiles -) - -// FirmwareUpgrade initiates a DFU update and returns the progress channel -func (c *Client) FirmwareUpgrade(upgType UpgradeType, files ...string) (<-chan DFUProgress, error) { - err := json.NewEncoder(c.conn).Encode(types.Request{ - Type: types.ReqTypeFwUpgrade, - Data: types.ReqDataFwUpgrade{ - Type: int(upgType), - Files: files, - }, - }) - if err != nil { - return nil, err - } - - c.dfuProgressCh = make(chan types.Response, 5) - - out := make(chan DFUProgress, 5) - go func() { - for res := range c.dfuProgressCh { - progress, err := decodeDFUProgress(res.Value) - if err != nil { - continue - } - out <- progress - } - }() - - return out, nil -} diff --git a/api/watch.go b/api/watch.go new file mode 100644 index 0000000..25ffb46 --- /dev/null +++ b/api/watch.go @@ -0,0 +1,144 @@ +package api + +import ( + "context" + "time" + + "go.arsenm.dev/infinitime" +) + +func (c *Client) WatchHeartRate() (<-chan uint8, func(), error) { + var id string + err := c.itdClient.Call( + context.Background(), + "WatchHeartRate", + nil, + &id, + ) + if err != nil { + return nil, nil, err + } + + outCh := make(chan uint8, 2) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + outCh <- val.(uint8) + } + }() + + doneFn := func() { + c.done(id) + close(c.srvVals[id]) + delete(c.srvVals, id) + } + + return outCh, doneFn, nil +} + +func (c *Client) WatchBatteryLevel() (<-chan uint8, func(), error) { + var id string + err := c.itdClient.Call( + context.Background(), + "WatchBatteryLevel", + nil, + &id, + ) + if err != nil { + return nil, nil, err + } + + outCh := make(chan uint8, 2) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + outCh <- val.(uint8) + } + }() + + doneFn := func() { + c.done(id) + close(c.srvVals[id]) + delete(c.srvVals, id) + } + + return outCh, doneFn, nil +} + +func (c *Client) WatchStepCount() (<-chan uint32, func(), error) { + var id string + err := c.itdClient.Call( + context.Background(), + "WatchStepCount", + nil, + &id, + ) + if err != nil { + return nil, nil, err + } + + outCh := make(chan uint32, 2) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + outCh <- val.(uint32) + } + }() + + doneFn := func() { + c.done(id) + close(c.srvVals[id]) + delete(c.srvVals, id) + } + + return outCh, doneFn, nil +} + +func (c *Client) WatchMotion() (<-chan infinitime.MotionValues, func(), error) { + var id string + err := c.itdClient.Call( + context.Background(), + "WatchMotion", + nil, + &id, + ) + if err != nil { + return nil, nil, err + } + + outCh := make(chan infinitime.MotionValues, 2) + go func() { + srvValCh, ok := c.srvVals[id] + for !ok { + time.Sleep(100 * time.Millisecond) + srvValCh, ok = c.srvVals[id] + } + + for val := range srvValCh { + outCh <- val.(infinitime.MotionValues) + } + }() + + doneFn := func() { + c.done(id) + close(c.srvVals[id]) + delete(c.srvVals, id) + } + + return outCh, doneFn, nil +} diff --git a/api/weather.go b/api/weather.go deleted file mode 100644 index 5d937c2..0000000 --- a/api/weather.go +++ /dev/null @@ -1,17 +0,0 @@ -package api - -import ( - "go.arsenm.dev/itd/internal/types" -) - -// UpdateWeather sends the update weather signal, -// immediately sending current weather data -func (c *Client) UpdateWeather() error { - _, err := c.request(types.Request{ - Type: types.ReqTypeWeatherUpdate, - }) - if err != nil { - return err - } - return nil -} diff --git a/cmd/itctl/firmware.go b/cmd/itctl/firmware.go index 913423b..cf2c12f 100644 --- a/cmd/itctl/firmware.go +++ b/cmd/itctl/firmware.go @@ -8,7 +8,6 @@ import ( "github.com/cheggaaa/pb/v3" "github.com/urfave/cli/v2" "go.arsenm.dev/itd/api" - "go.arsenm.dev/itd/internal/types" ) func fwUpgrade(c *cli.Context) error { @@ -19,11 +18,11 @@ func fwUpgrade(c *cli.Context) error { // Get relevant data struct if c.String("archive") != "" { // Get archive data struct - upgType = types.UpgradeTypeArchive + upgType = api.UpgradeTypeArchive files = []string{c.String("archive")} } else if c.String("init-packet") != "" && c.String("firmware") != "" { // Get files data struct - upgType = types.UpgradeTypeFiles + upgType = api.UpgradeTypeFiles files = []string{c.String("init-packet"), c.String("firmware")} } else { return cli.Exit("Upgrade command requires either archive or init packet and firmware.", 1) @@ -43,9 +42,9 @@ func fwUpgrade(c *cli.Context) error { // Set total bytes in progress bar bar.SetTotal(event.Total) // Set amount of bytes received in progress bar - bar.SetCurrent(event.Received) + bar.SetCurrent(int64(event.Received)) // If transfer finished, break - if event.Sent == event.Total { + if int64(event.Sent) == event.Total { break } } diff --git a/cmd/itctl/fs.go b/cmd/itctl/fs.go index 2f90d9c..88f2591 100644 --- a/cmd/itctl/fs.go +++ b/cmd/itctl/fs.go @@ -76,7 +76,7 @@ func fsRead(c *cli.Context) error { } } - progress, err := client.ReadFile(path, c.Args().Get(0)) + progress, err := client.Download(path, c.Args().Get(0)) if err != nil { return err } @@ -91,12 +91,8 @@ func fsRead(c *cli.Context) error { bar.SetTotal(int64(event.Total)) // Set amount of bytes sent in progress bar bar.SetCurrent(int64(event.Sent)) - // If transfer finished, break - if event.Done { - bar.Finish() - break - } } + bar.Finish() if c.Args().Get(1) == "-" { io.Copy(os.Stdout, tmpFile) @@ -148,7 +144,7 @@ func fsWrite(c *cli.Context) error { defer os.Remove(path) } - progress, err := client.WriteFile(path, c.Args().Get(1)) + progress, err := client.Upload(c.Args().Get(1), path) if err != nil { return err } @@ -163,11 +159,6 @@ func fsWrite(c *cli.Context) error { bar.SetTotal(int64(event.Total)) // Set amount of bytes sent in progress bar bar.SetCurrent(int64(event.Sent)) - // If transfer finished, break - if event.Done { - bar.Finish() - break - } } return nil diff --git a/cmd/itctl/set.go b/cmd/itctl/set.go index f0590ae..914a3bb 100644 --- a/cmd/itctl/set.go +++ b/cmd/itctl/set.go @@ -13,7 +13,7 @@ func setTime(c *cli.Context) error { } if c.Args().Get(0) == "now" { - return client.SetTimeNow() + return client.SetTime(time.Now()) } else { parsed, err := time.Parse(time.RFC3339, c.Args().Get(0)) if err != nil { diff --git a/cmd/itctl/update.go b/cmd/itctl/update.go index bda3bcf..6f8e04f 100644 --- a/cmd/itctl/update.go +++ b/cmd/itctl/update.go @@ -3,5 +3,5 @@ package main import "github.com/urfave/cli/v2" func updateWeather(c *cli.Context) error { - return client.UpdateWeather() + return client.WeatherUpdate() } diff --git a/cmd/itgui/time.go b/cmd/itgui/time.go index b093a3b..e36cefa 100644 --- a/cmd/itgui/time.go +++ b/cmd/itgui/time.go @@ -49,7 +49,7 @@ func timeTab(parent fyne.Window, client *api.Client) *fyne.Container { func setTime(client *api.Client, current bool, t ...time.Time) error { var err error if current { - err = client.SetTimeNow() + err = client.SetTime(time.Now()) } else { err = client.SetTime(t[0]) } diff --git a/cmd/itgui/upgrade.go b/cmd/itgui/upgrade.go index f9b31db..130c04d 100644 --- a/cmd/itgui/upgrade.go +++ b/cmd/itgui/upgrade.go @@ -11,7 +11,6 @@ import ( "fyne.io/fyne/v2/storage" "fyne.io/fyne/v2/widget" "go.arsenm.dev/itd/api" - "go.arsenm.dev/itd/internal/types" ) func upgradeTab(parent fyne.Window, client *api.Client) *fyne.Container { @@ -119,10 +118,10 @@ func upgradeTab(parent fyne.Window, client *api.Client) *fyne.Container { // Get appropriate upgrade type and file paths switch upgradeTypeSelect.Selected { case "Archive": - fwUpgType = types.UpgradeTypeArchive + fwUpgType = api.UpgradeTypeArchive files = append(files, archivePath) case "Files": - fwUpgType = types.UpgradeTypeFiles + fwUpgType = api.UpgradeTypeFiles files = append(files, initPktPath, firmwarePath) } @@ -144,7 +143,7 @@ func upgradeTab(parent fyne.Window, client *api.Client) *fyne.Container { // Refresh progress bar progressBar.Refresh() // If transfer finished, break - if event.Sent == event.Total { + if int64(event.Sent) == event.Total { break } } diff --git a/go.mod b/go.mod index 80f30d1..959fd59 100644 --- a/go.mod +++ b/go.mod @@ -7,31 +7,25 @@ require ( github.com/VividCortex/ewma v1.2.0 // indirect github.com/cheggaaa/pb/v3 v3.0.8 github.com/cpuguy83/go-md2man/v2 v2.0.0 // indirect - github.com/fatih/color v1.13.0 // indirect - github.com/fsnotify/fsnotify v1.5.1 // indirect github.com/gen2brain/dlgs v0.0.0-20211108104213-bade24837f0b github.com/go-gl/gl v0.0.0-20211210172815-726fda9656d6 // indirect github.com/go-gl/glfw/v3.3/glfw v0.0.0-20211204153444-caad923f49f4 // indirect github.com/godbus/dbus/v5 v5.0.6 github.com/google/uuid v1.3.0 - github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 // indirect github.com/knadh/koanf v1.4.0 - github.com/mattn/go-colorable v0.1.12 // indirect github.com/mattn/go-isatty v0.0.14 github.com/mattn/go-runewidth v0.0.13 // indirect - github.com/mitchellh/mapstructure v1.4.3 github.com/mozillazg/go-pinyin v0.19.0 github.com/pelletier/go-toml v1.9.4 // indirect github.com/rs/zerolog v1.26.0 github.com/sirupsen/logrus v1.8.1 // indirect + github.com/smallnest/rpcx v1.7.4 github.com/srwiley/oksvg v0.0.0-20211120171407-1837d6608d8c // indirect github.com/srwiley/rasterx v0.0.0-20210519020934-456a8d69b780 // indirect github.com/urfave/cli/v2 v2.3.0 + github.com/vmihailenco/msgpack/v5 v5.3.5 github.com/yuin/goldmark v1.4.4 // indirect go.arsenm.dev/infinitime v0.0.0-20220416112421-b7a50271bece golang.org/x/image v0.0.0-20211028202545-6944b10bf410 // indirect - golang.org/x/net v0.0.0-20211209124913-491a49abca63 // indirect golang.org/x/text v0.3.7 - gopkg.in/yaml.v2 v2.4.0 // indirect - gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect ) diff --git a/go.sum b/go.sum index af5d53f..7328902 100644 --- a/go.sum +++ b/go.sum @@ -1,15 +1,39 @@ cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.31.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +cloud.google.com/go v0.37.0/go.mod h1:TS1dMSSfndXH133OKGwekG838Om/cQT0BUHV3HcBgoo= +dmitri.shuralyov.com/app/changes v0.0.0-20180602232624-0a106ad413e3/go.mod h1:Yl+fi1br7+Rr3LqpNJf1/uxUdtRUV+Tnj0o93V2B9MU= +dmitri.shuralyov.com/html/belt v0.0.0-20180602232347-f7d459c86be0/go.mod h1:JLBrvjyP0v+ecvNYvCpyZgu5/xkfAUhi6wJj28eUfSU= +dmitri.shuralyov.com/service/change v0.0.0-20181023043359-a85b471d5412/go.mod h1:a1inKt/atXimZ4Mv927x+r7UpyzRUf4emIoiiSC2TN4= +dmitri.shuralyov.com/state v0.0.0-20180228185332-28bcc343414c/go.mod h1:0PRwlb0D6DFvNNtx+9ybjezNCa8XF0xaYcETyp6rHWU= fyne.io/fyne/v2 v2.1.2 h1:avp9CvLAUdvE7fDMtH1tVKyjxEWHWcpow6aI6L7Kvvw= fyne.io/fyne/v2 v2.1.2/go.mod h1:p+E/Dh+wPW8JwR2DVcsZ9iXgR9ZKde80+Y+40Is54AQ= +git.apache.org/thrift.git v0.0.0-20180902110319-2566ecd5d999/go.mod h1:fPE2ZNJGynbRyZ4dJvy6G277gSllfV2HJqblrnkyeyg= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/BurntSushi/toml v0.4.1/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= +github.com/ChimeraCoder/gojson v1.1.0/go.mod h1:nYbTQlu6hv8PETM15J927yM0zGj3njIldp72UT1MqSw= +github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= github.com/Kodeworks/golang-image-ico v0.0.0-20141118225523-73f0f4cfade9/go.mod h1:7uhhqiBaR4CpN0k9rMjOtjpcfGd6DG2m04zQxKnWQ0I= github.com/VividCortex/ewma v1.1.1/go.mod h1:2Tkkvm3sRDVXaiyucHiACn4cqf7DpdyLvmxzcbUokwA= github.com/VividCortex/ewma v1.2.0 h1:f58SaIzcDXrSy3kWaHNvuJgJ3Nmz59Zji6XoJR/q1ow= github.com/VividCortex/ewma v1.2.0/go.mod h1:nz4BbCtbLyFDeC9SUHbtcT5644juEuWfUAUnGx7j5l4= github.com/akavel/rsrc v0.8.0/go.mod h1:uLoCtb9J+EyAqh+26kdrTgmzRBFPGOolLWKpdxkKq+c= +github.com/akutz/memconn v0.1.0 h1:NawI0TORU4hcOMsMr11g7vwlCdkYeLKXBcxWu2W/P8A= +github.com/akutz/memconn v0.1.0/go.mod h1:Jo8rI7m0NieZyLI5e2CDlRdRqRRB4S7Xp77ukDjH+Fw= +github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= +github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= +github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239/go.mod h1:2FmKhYUyUczH0OGQWaF5ceTx0UBShxjsH6f8oGKYe2c= +github.com/apache/thrift v0.16.0 h1:qEy6UW60iVOlUy+b9ZR0d5WzUWYGOo4HfopoyBaNmoY= +github.com/apache/thrift v0.16.0/go.mod h1:PHK3hniurgQaNMZYaCLEqXKsYK8upmhPbmdP2FXSqgU= +github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= +github.com/armon/go-metrics v0.3.6/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= +github.com/armon/go-metrics v0.3.10 h1:FR+drcQStOe+32sYyJYyZ7FIdgoGGBnwLl+flodp8Uo= +github.com/armon/go-metrics v0.3.10/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= +github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= github.com/aws/aws-sdk-go-v2 v1.9.2/go.mod h1:cK/D0BBs0b/oWPIcX/Z/obahJK1TT7IPVjy53i/mX/4= github.com/aws/aws-sdk-go-v2/config v1.8.3/go.mod h1:4AEiLtAb8kLs7vgw2ZV3p2VZ1+hBavOc84hqxVNpCyw= github.com/aws/aws-sdk-go-v2/credentials v1.4.3/go.mod h1:FNNC6nQZQUuyhq5aE5c7ata8o9e4ECGmS4lAXC7o1mQ= @@ -20,10 +44,32 @@ github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.3.2/go.mod h1:72H github.com/aws/aws-sdk-go-v2/service/sso v1.4.2/go.mod h1:NBvT9R1MEF+Ud6ApJKM0G+IkPchKS7p7c2YPKwHmBOk= github.com/aws/aws-sdk-go-v2/service/sts v1.7.2/go.mod h1:8EzeIqfWt2wWT4rJVu3f21TfrhJ8AEMzVybRNSb/b4g= github.com/aws/smithy-go v1.8.0/go.mod h1:SObp3lf9smib00L/v3U2eAKG8FyQ7iLrJnQiAmR5n+E= +github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= +github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= +github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= +github.com/bradfitz/go-smtpd v0.0.0-20170404230938-deb6d6237625/go.mod h1:HYsPBTaaSFSlLx/70C2HPIMNZpVV8+vt/A+FMnYP11g= +github.com/buger/jsonparser v0.0.0-20181115193947-bf1c66bbce23/go.mod h1:bbYlZJ7hK1yFx9hf58LP0zeX7UjIGs20ufpu3evjr+s= +github.com/cenk/backoff v2.2.1+incompatible h1:djdFT7f4gF2ttuzRKPbMOWgZajgesItGLwG5FTQKmmE= +github.com/cenk/backoff v2.2.1+incompatible/go.mod h1:7FtoeaSnHoZnmZzz47cM35Y9nSW7tNyaidugnHTaFDE= +github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4= +github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= +github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= +github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/cheekybits/genny v1.0.0 h1:uGGa4nei+j20rOSeDeP5Of12XVm7TGUd4dJA9RDitfE= +github.com/cheekybits/genny v1.0.0/go.mod h1:+tQajlRqAUrPI7DOSpB0XAqZYtQakVtB7wXkRAgjxjQ= github.com/cheggaaa/pb/v3 v3.0.8 h1:bC8oemdChbke2FHIIGy9mn4DPJ2caZYQnfbRqwmdCoA= github.com/cheggaaa/pb/v3 v3.0.8/go.mod h1:UICbiLec/XO6Hw6k+BHEtHeQFzzBH4i2/qk/ow1EJTA= +github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= +github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= +github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= +github.com/circonus-labs/circonus-gometrics v2.3.1+incompatible/go.mod h1:nmEj6Dob7S7YxXgwXpfOuvO54S+tGdZdw9fuRZt25Ag= +github.com/circonus-labs/circonusllhist v0.1.3/go.mod h1:kMXHVDlOchFAehlya5ePtbp5jckzBHf4XRpQvBOLI+I= github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= +github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= +github.com/coreos/go-systemd v0.0.0-20181012123002-c6f51f82210d/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/cpuguy83/go-md2man/v2 v2.0.0 h1:EoUDS0afbrsXAZ9YQ9jdu/mZ2sXgT1/2yyNng4PGlyM= @@ -31,14 +77,31 @@ github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsr github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/dgryski/go-jump v0.0.0-20170409065014-e1f439676b57/go.mod h1:4hKCXuwrJoYvHZxJ86+bRVTOMyJ0Ej+RqfSm8mHi6KA= +github.com/dgryski/go-jump v0.0.0-20211018200510-ba001c3ffce0 h1:0wH6nO9QEa02Qx8sIQGw6ieKdz+BXjpccSOo9vXNl4U= +github.com/dgryski/go-jump v0.0.0-20211018200510-ba001c3ffce0/go.mod h1:4hKCXuwrJoYvHZxJ86+bRVTOMyJ0Ej+RqfSm8mHi6KA= +github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= +github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= +github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= +github.com/edwingeng/doublejump v0.0.0-20210724020454-c82f1bcb3280 h1:t04U/MQCFMNbKQ1YdqsZe8MQIgYB4gGOX87CQIrStqw= +github.com/edwingeng/doublejump v0.0.0-20210724020454-c82f1bcb3280/go.mod h1:+RhzcuwqMSuWjEJrOKR19t+Wc8Ca7OuUYQ//CReOIwI= +github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= +github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= +github.com/facebookgo/clock v0.0.0-20150410010913-600d898af40a h1:yDWHCSQ40h88yih2JAcL6Ls/kVkSE8GFACTGVnMPruw= +github.com/facebookgo/clock v0.0.0-20150410010913-600d898af40a/go.mod h1:7Ga40egUymuWXxAe151lTNnCv97MddSOVsjpPPkityA= github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= +github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= github.com/fatih/color v1.10.0/go.mod h1:ELkj/draVOlAH/xkhN6mQ50Qd0MPOk5AAr3maGEBuJM= github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w= github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= github.com/fatih/structs v1.1.0 h1:Q7juDM0QtcnhCpeyLGQKyg4TOIghuNXrkL32pHAUMxo= github.com/fatih/structs v1.1.0/go.mod h1:9NiDSp5zOcgEDl+j00MP/WkGVPOlPRLejGD8Ga6PJ7M= +github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= +github.com/francoispqt/gojay v1.2.13/go.mod h1:ehT5mTG4ua4581f1++1WLG0vPdaA9HaiDsoyrBGkyDY= github.com/fredbi/uri v0.0.0-20181227131451-3dcfdacbaaf3 h1:FDqhDm7pcsLhhWl1QtD8vlzI4mm59llRvNzrFg6/LAA= github.com/fredbi/uri v0.0.0-20181227131451-3dcfdacbaaf3/go.mod h1:CzM2G82Q9BDUvMTGHnXf/6OExw/Dz2ivDj48nVg7Lg8= +github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= github.com/fsnotify/fsnotify v1.5.1 h1:mZcQUHVQUQWoPXXtuf9yuEXKudkV2sx1E06UadKWpgI= github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= @@ -46,89 +109,254 @@ github.com/fxamacker/cbor/v2 v2.4.0 h1:ri0ArlOR+5XunOP8CRUowT0pSJOwhW098ZCUyskZD github.com/fxamacker/cbor/v2 v2.4.0/go.mod h1:TA1xS00nchWmaBnEIxPSE5oHLuJBAVvqrtAnWBwBCVo= github.com/gen2brain/dlgs v0.0.0-20211108104213-bade24837f0b h1:M0/hjawi9ur15zpqL/h66ga87jlYA7iAuZ4HC6ak08k= github.com/gen2brain/dlgs v0.0.0-20211108104213-bade24837f0b/go.mod h1:/eFcjDXaU2THSOOqLxOPETIbHETnamk8FA/hMjhg/gU= +github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/gliderlabs/ssh v0.1.1/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0= +github.com/go-errors/errors v1.0.1/go.mod h1:f4zRHt4oKfwPJE5k8C9vpYG+aDHdBFUsgrm6/TyX73Q= github.com/go-gl/gl v0.0.0-20210813123233-e4099ee2221f/go.mod h1:wjpnOv6ONl2SuJSxqCPVaPZibGFdSci9HFocT9qtVYM= github.com/go-gl/gl v0.0.0-20211210172815-726fda9656d6 h1:zDw5v7qm4yH7N8C8uWd+8Ii9rROdgWxQuGoJ9WDXxfk= github.com/go-gl/gl v0.0.0-20211210172815-726fda9656d6/go.mod h1:9YTyiznxEY1fVinfM7RvRcjRHbw2xLBJ3AAGIT0I4Nw= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20211024062804-40e447a793be/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20211204153444-caad923f49f4 h1:KgfIc81yNEUKNAsF+Mt3C1Cl+iQqKF1r7nWEKzL0c2Y= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20211204153444-caad923f49f4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= +github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= +github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-ldap/ldap v3.0.2+incompatible/go.mod h1:qfd9rJvER9Q0/D/Sqn1DfHRoBp40uXYvFoEVrNEPqRc= +github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= +github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= +github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/logr v1.2.3 h1:2DntVwHkVopvECVRSlL5PSo9eG+cAkDCuckLubN+rq0= +github.com/go-logr/logr v1.2.3/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= +github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= +github.com/go-ping/ping v0.0.0-20211130115550-779d1e919534 h1:dhy9OQKGBh4zVXbjwbxxHjRxMJtLXj3zfgpBYQaR4Q4= +github.com/go-ping/ping v0.0.0-20211130115550-779d1e919534/go.mod h1:xIFjORFzTxqIV/tDVGO4eDy/bLuSyawEeojSm3GfRGk= +github.com/go-redis/redis/v8 v8.8.2/go.mod h1:F7resOH5Kdug49Otu24RjHWwgK7u9AmtqWMnCV1iP5Y= +github.com/go-redis/redis/v8 v8.11.4/go.mod h1:2Z2wHZXdQpCDXEGzqMockDpNyYvi2l4Pxt6RJr792+w= +github.com/go-redis/redis/v8 v8.11.5 h1:AcZZR7igkdvfVmQTPnu9WE37LRrO/YrBH5zWyjDC0oI= +github.com/go-redis/redis/v8 v8.11.5/go.mod h1:gREzHqY1hg6oD9ngVRbLStwAWKhA0FEgq8Jd4h5lpwo= +github.com/go-redis/redis_rate/v9 v9.1.2/go.mod h1:oam2de2apSgRG8aJzwJddXbNu91Iyz1m8IKJE2vpvlQ= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0 h1:p104kn46Q8WdvHunIJ9dAyjPVtrBPhSr3KT2yUst43I= +github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= github.com/go-test/deep v1.0.2-0.20181118220953-042da051cf31/go.mod h1:wGDj63lr65AM2AQyKZd/NYHGb0R+1RLqB8NKt3aSFNA= github.com/godbus/dbus/v5 v5.0.3/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/godbus/dbus/v5 v5.0.6 h1:mkgN1ofwASrYnJ5W6U/BxG15eXXXjirgZc7CLqkcaro= github.com/godbus/dbus/v5 v5.0.6/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= +github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/goki/freetype v0.0.0-20181231101311-fa8a33aabaff h1:W71vTCKoxtdXgnm1ECDFkfQnpdqAO00zzGXLA5yaEX8= github.com/goki/freetype v0.0.0-20181231101311-fa8a33aabaff/go.mod h1:wfqRWLHRBsRgkp5dmbG56SA0DmVtwrF5N3oPdI8t+Aw= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= +github.com/golang/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:tluoj9z5200jBnyusfRPU2LqT6J+DAorxEvtC7LHB+E= github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= +github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8= +github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= +github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= +github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= +github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= +github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= +github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= +github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= +github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= +github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= +github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= +github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c h1:964Od4U6p2jUkFxvCydnIczKteheJEzHRToSGK3Bnlw= +github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= +github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.7 h1:81/ik6ipDQS2aGcBfIN5dHDB36BwrStyeAQquSYCV4o= +github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= +github.com/google/go-github v17.0.0+incompatible/go.mod h1:zLgOLi98H3fifZn+44m+umXrS52loVEgC2AApnigrVQ= +github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= +github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/googleapis/gax-go v2.0.0+incompatible/go.mod h1:SFVmujtThgffbyetf+mdk2eWhX2bMyUtNHzFKcPA9HY= +github.com/googleapis/gax-go/v2 v2.0.3/go.mod h1:LLvjysVCY1JZeum8Z6l8qUty8fiNwE08qbEPm1M08qg= github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8= github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= +github.com/grandcat/zeroconf v1.0.0 h1:uHhahLBKqwWBV6WZUDAT71044vwOTL+McW0mBJvo6kE= +github.com/grandcat/zeroconf v1.0.0/go.mod h1:lTKmG1zh86XyCoUeIHSA4FJMBwCJiQmGfcP2PdzytEs= +github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA= +github.com/grpc-ecosystem/grpc-gateway v1.5.0/go.mod h1:RSKVYQBd5MCa4OVpNdGskqpgL2+G+NZTnrVHpWWfpdw= +github.com/hashicorp/consul/api v1.8.1/go.mod h1:sDjTOq0yUyv5G4h+BqSea7Fn6BU+XbolEz1952UB+mk= +github.com/hashicorp/consul/api v1.12.0 h1:k3y1FYv6nuKyNTqj6w9gXOx5r5CfLj/k/euUeBXj1OY= +github.com/hashicorp/consul/api v1.12.0/go.mod h1:6pVBMo0ebnYdt2S3H87XhekM/HHrUoTD2XXb/VrZVy0= +github.com/hashicorp/consul/sdk v0.7.0/go.mod h1:fY08Y9z5SvJqevyZNy6WWPXiG3KwBPAvlcdx16zZ0fM= +github.com/hashicorp/consul/sdk v0.8.0 h1:OJtKBtEjboEZvG6AOUdh4Z1Zbyu0WcxQ0qatRrZHTVU= +github.com/hashicorp/consul/sdk v0.8.0/go.mod h1:GBvyrGALthsZObzUGsfgHZQDXjg4lOjagTIwIR1vPms= github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= +github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I= +github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ= +github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= github.com/hashicorp/go-hclog v0.0.0-20180709165350-ff2cf002a8dd/go.mod h1:9bjs9uLqI8l75knNv3lV1kA55veR+WUPSiKIWcQHudI= github.com/hashicorp/go-hclog v0.8.0/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ= +github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-hclog v0.16.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-hclog v1.2.0 h1:La19f8d7WIlm4ogzNHB0JGqs5AUDAZ2UfCY4sJXcJdM= +github.com/hashicorp/go-hclog v1.2.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-immutable-radix v1.3.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-immutable-radix v1.3.1 h1:DKHmCUm2hRBK510BaiZlwvpD40f8bJFeZnpfm2KLowc= +github.com/hashicorp/go-immutable-radix v1.3.1/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-msgpack v0.5.3 h1:zKjpN5BK/P5lMYrLmBHdBULWbJ0XpYR+7NGzqkZzoD4= +github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= +github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= +github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo= +github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM= github.com/hashicorp/go-plugin v1.0.1/go.mod h1:++UyYGoz3o5w9ZzAdZxtQKrWWP+iqPBn3cQptSMzBuY= +github.com/hashicorp/go-retryablehttp v0.5.3/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs= github.com/hashicorp/go-retryablehttp v0.5.4/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs= github.com/hashicorp/go-rootcerts v1.0.1/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= +github.com/hashicorp/go-rootcerts v1.0.2 h1:jzhAVGtqPKbwpyCPELlgNWhE1znq+qwJtW5Oi2viEzc= +github.com/hashicorp/go-rootcerts v1.0.2/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= +github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= +github.com/hashicorp/go-sockaddr v1.0.2 h1:ztczhD1jLxIRjVejw8gFomI1BQZOe2WoVOu0SyteCQc= github.com/hashicorp/go-sockaddr v1.0.2/go.mod h1:rB4wwRAUzs07qva3c5SdrY/NEtAUjGlgmH/UkBUC97A= +github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-uuid v1.0.1 h1:fv1ep09latC32wFoVwnqcnKJGnMSdBanPczbHAYm1BE= github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= github.com/hashicorp/go-version v1.1.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/golang-lru v0.5.4 h1:YDjusn29QI/Das2iO9M0BHnIbxPeyuCHsjMW+lJfyTc= +github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= +github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= +github.com/hashicorp/mdns v1.0.1/go.mod h1:4gW7WsVCke5TE7EPeYliwHlRUyBtfCwuFwuMg2DmyNY= +github.com/hashicorp/mdns v1.0.4/go.mod h1:mtBihi+LeNXGtG8L9dX59gAEa12BDtBQSp4v/YAJqrc= +github.com/hashicorp/memberlist v0.2.2/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/memberlist v0.3.0 h1:8+567mCcFDnS5ADl7lrpxPMWiFCElyUEeW0gtj34fMA= +github.com/hashicorp/memberlist v0.3.0/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/serf v0.9.5/go.mod h1:UWDWwZeL5cuWDJdl0C6wrvrUwEqtQ4ZKBKKENpqIUyk= +github.com/hashicorp/serf v0.9.6/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= +github.com/hashicorp/serf v0.9.7 h1:hkdgbqizGQHuU5IPqYM1JdSMV8nKfpuOnZYXssk9muY= +github.com/hashicorp/serf v0.9.7/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= github.com/hashicorp/vault/api v1.0.4/go.mod h1:gDcqh3WGcR1cpF5AJz/B1UFheUEneMoIospckxBxk6Q= github.com/hashicorp/vault/sdk v0.1.13/go.mod h1:B+hVj7TpuQY1Y/GPbCpffmgd+tSEwvhkWnjtSYCaS2M= github.com/hashicorp/yamux v0.0.0-20180604194846-3520598351bb/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM= github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM= +github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= +github.com/influxdata/influxdb1-client v0.0.0-20220302092344-a9ab5670611c/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= github.com/jackmordaunt/icns v0.0.0-20181231085925-4f16af745526/go.mod h1:UQkeMHVoNcyXYq9otUupF7/h/2tmHlhrS2zw7ZVvUqc= +github.com/jellevandenhooff/dkim v0.0.0-20150330215556-f50fe3d243e1/go.mod h1:E0B/fFc00Y+Rasa88328GlI/XbtyysCtTHZS8h7IrBU= github.com/jmespath/go-jmespath v0.4.0/go.mod h1:T8mJZnbsbmF+m6zOOFylbeCJqk5+pHWvzYPziyZiYoo= github.com/jmespath/go-jmespath/internal/testify v1.5.1/go.mod h1:L3OGu8Wl2/fWfCI6z80xFu9LTZmf1ZRjMHUOPmWr69U= github.com/joho/godotenv v1.3.0 h1:Zjp+RcGpHhGlrMbJzXTrZZPrWj+1vfm90La1wgB6Bhc= github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg= github.com/josephspurrier/goversioninfo v0.0.0-20200309025242-14b0ab84c6ca/go.mod h1:eJTEwMjXb7kZ633hO3Ln9mBUCOjX2+FlTljvpl9SYdE= +github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= +github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= +github.com/juju/ratelimit v1.0.1 h1:+7AIFJVQ0EQgq/K9+0Krm7m530Du7tIz0METWzN0RgY= +github.com/juju/ratelimit v1.0.1/go.mod h1:qapgC/Gy+xNh9UxzV13HGGl/6UXNN+ct+vwSgWNm/qk= +github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= +github.com/julienschmidt/httprouter v1.3.0 h1:U0609e9tgbseu3rBINet9P48AI/D3oJs4dN7jwJOQ1U= +github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= +github.com/kavu/go_reuseport v1.5.0 h1:UNuiY2OblcqAtVDE8Gsg1kZz8zbBWg907sP1ceBV+bk= +github.com/kavu/go_reuseport v1.5.0/go.mod h1:CG8Ee7ceMFSMnx/xr25Vm0qXaj2Z4i5PWoUx+JZ5/CU= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= +github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/klauspost/cpuid/v2 v2.0.6/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= +github.com/klauspost/cpuid/v2 v2.0.12 h1:p9dKCg8i4gmOxtv35DvrYoWqYzQrvEVdjQ762Y0OqZE= +github.com/klauspost/cpuid/v2 v2.0.12/go.mod h1:g2LTdtYhdyuGPqyWyv7qRAmj1WBqxuObKfj5c0PQa7c= +github.com/klauspost/reedsolomon v1.9.16 h1:mR0AwphBwqFv/I3B9AHtNKvzuowI1vrj8/3UX4XRmHA= +github.com/klauspost/reedsolomon v1.9.16/go.mod h1:eqPAcE7xar5CIzcdfwydOEdcmchAKAP/qs14y4GCBOk= github.com/knadh/koanf v1.4.0 h1:/k0Bh49SqLyLNfte9r6cvuZWrApOQhglOmhIU3L/zDw= github.com/knadh/koanf v1.4.0/go.mod h1:1cfH5223ZeZUOs8FU2UdTmaNfHpqgtjV0+NHjRO43gs= +github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/pty v1.1.3/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/lucas-clemente/quic-go v0.24.0/go.mod h1:paZuzjXCE5mj6sikVLMvqXk8lJV2AsqtJ6bDhjEfxx0= +github.com/lucas-clemente/quic-go v0.27.0 h1:v6WY87q9zD4dKASbG8hy/LpzAVNzEQzw8sEIeloJsc4= +github.com/lucas-clemente/quic-go v0.27.0/go.mod h1:AzgQoPda7N+3IqMMMkywBKggIFo2KT6pfnlrQ2QieeI= github.com/lucor/goinfo v0.0.0-20210802170112-c078a2b0f08b/go.mod h1:PRq09yoB+Q2OJReAmwzKivcYyremnibWGbK7WfftHzc= +github.com/lunixbochs/vtclean v1.0.0/go.mod h1:pHhQNgMf3btfWnGBVipUOjRYhoOsdGqdm/+2c2E2WMI= +github.com/mailru/easyjson v0.0.0-20190312143242-1de009706dbe/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/marten-seemann/qpack v0.2.1/go.mod h1:F7Gl5L1jIgN1D11ucXefiuJS9UMVP2opoCp2jDKb7wc= +github.com/marten-seemann/qtls-go1-15 v0.1.4/go.mod h1:GyFwywLKkRt+6mfU99csTEY1joMZz5vmB1WNZH3P81I= +github.com/marten-seemann/qtls-go1-16 v0.1.4/go.mod h1:gNpI2Ol+lRS3WwSOtIUUtRwZEQMXjYK+dQSBFbethAk= +github.com/marten-seemann/qtls-go1-16 v0.1.5 h1:o9JrYPPco/Nukd/HpOHMHZoBDXQqoNtUCmny98/1uqQ= +github.com/marten-seemann/qtls-go1-16 v0.1.5/go.mod h1:gNpI2Ol+lRS3WwSOtIUUtRwZEQMXjYK+dQSBFbethAk= +github.com/marten-seemann/qtls-go1-17 v0.1.0/go.mod h1:fz4HIxByo+LlWcreM4CZOYNuz3taBQ8rN2X6FqvaWo8= +github.com/marten-seemann/qtls-go1-17 v0.1.1 h1:DQjHPq+aOzUeh9/lixAGunn6rIOQyWChPSI4+hgW7jc= +github.com/marten-seemann/qtls-go1-17 v0.1.1/go.mod h1:C2ekUKcDdz9SDWxec1N/MvcXBpaX9l3Nx67XaR84L5s= +github.com/marten-seemann/qtls-go1-18 v0.1.1 h1:qp7p7XXUFL7fpBvSS1sWD+uSqPvzNQK43DH+/qEkj0Y= +github.com/marten-seemann/qtls-go1-18 v0.1.1/go.mod h1:mJttiymBAByA49mhlNZZGrH5u1uXYZJ+RW28Py7f4m4= github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= +github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= +github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-colorable v0.1.12 h1:jF+Du6AlPIjs2BiUiQlKOX0rt3SujHxPnksPKZbaA40= github.com/mattn/go-colorable v0.1.12/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= +github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= +github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= github.com/mattn/go-runewidth v0.0.12/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk= github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU= github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= +github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= +github.com/microcosm-cc/bluemonday v1.0.1/go.mod h1:hsXNsILzKxV+sX77C5b8FSuKF00vh2OMYv+xgHpAMF4= +github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= +github.com/miekg/dns v1.1.26/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= +github.com/miekg/dns v1.1.27/go.mod h1:KNUDUusw/aVsxyTYZM1oqvCicbwhgbNgztCETuNZ7xM= +github.com/miekg/dns v1.1.41/go.mod h1:p6aan82bvRIyn+zDIv9xYNUpwa73JcSh9BKwknJysuI= +github.com/miekg/dns v1.1.48 h1:Ucfr7IIVyMBz4lRE8qmGUuZ4Wt3/ZGu9hmcMT3Uu4tQ= +github.com/miekg/dns v1.1.48/go.mod h1:e3IlAVfNqAllflbibAZEWOXOQ+Ynzk/dDozDxY7XnME= github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= +github.com/mitchellh/cli v1.1.0/go.mod h1:xcISNoH86gajksDmfB23e/pu+B+GeFRMYmoHXxx3xhI= github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw= github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= +github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= +github.com/mitchellh/go-testing-interface v1.0.0 h1:fzU/JVNcaqHQEcVFAKeR41fkiLdIPrefOvVG1VZ96U0= github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= github.com/mitchellh/go-wordwrap v1.0.0/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo= +github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= github.com/mitchellh/mapstructure v1.4.1/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= github.com/mitchellh/mapstructure v1.4.3 h1:OVowDSCllw/YjdLkam3/sm7wEtOy59d8ndGgCcyj8cs= @@ -136,42 +364,143 @@ github.com/mitchellh/mapstructure v1.4.3/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RR github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ= github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= github.com/mozillazg/go-pinyin v0.19.0 h1:p+J8/kjJ558KPvVGYLvqBhxf8jbZA2exSLCs2uUVN8c= github.com/mozillazg/go-pinyin v0.19.0/go.mod h1:iR4EnMMRXkfpFVV5FMi4FNB6wGq9NV6uDWbUuPhP4Yc= github.com/muka/go-bluetooth v0.0.0-20220219050759-674a63b8741a h1:fnzS9RRQW8B5AgNCxkN0vJ/AoX+Xfqk3sAYon3iVrzA= github.com/muka/go-bluetooth v0.0.0-20220219050759-674a63b8741a/go.mod h1:dMCjicU6vRBk34dqOmIZm0aod6gUwZXOXzBROqGous0= +github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/neelance/astrewrite v0.0.0-20160511093645-99348263ae86/go.mod h1:kHJEU3ofeGjhHklVoIGuVj85JJwZ6kWPaJwCIxgnFmo= +github.com/neelance/sourcemap v0.0.0-20151028013722-8c68805598ab/go.mod h1:Qr6/a/Q4r9LP1IltGz7tA7iOK1WonHEYhu1HRBA7ZiM= github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/npillmayer/nestext v0.1.3/go.mod h1:h2lrijH8jpicr25dFY+oAJLyzlya6jhnuG+zWp9L0Uk= +github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= +github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= +github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA= +github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= +github.com/onsi/ginkgo v1.14.0/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY= +github.com/onsi/ginkgo v1.15.0/go.mod h1:hF8qUzuuC8DJGygJH3726JnCZX4MYbRB8yFfISqnKUg= +github.com/onsi/ginkgo v1.16.2/go.mod h1:CObGmKUOKaSC0RjmoAK7tKyn4Azo5P2IWuoMnvwxz1E= +github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= +github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= +github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= +github.com/onsi/ginkgo/v2 v2.0.0/go.mod h1:vw5CSIxN1JObi/U8gcbwft7ZxR2dgaR70JSE3/PpL4c= +github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= +github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= +github.com/onsi/gomega v1.10.5/go.mod h1:gza4q3jKQJijlu05nKWRCW/GavJumGt8aNRxWg7mt48= +github.com/onsi/gomega v1.13.0/go.mod h1:lRk9szgn8TxENtWd0Tp4c3wjlRfMTMH27I+3Je41yGY= +github.com/onsi/gomega v1.16.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= +github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= +github.com/onsi/gomega v1.18.1 h1:M1GfJqGRrBrrGGsbxzV5dqM2U2ApXefZCQpkukxYRLE= +github.com/onsi/gomega v1.18.1/go.mod h1:0q+aL8jAiMXy9hbwj2mr5GziHiwhAIQpFmmtT5hitRs= +github.com/openzipkin/zipkin-go v0.1.1/go.mod h1:NtoC/o8u3JlF1lSlyPNswIbeQH9bJTmOf0Erfk+hxe8= +github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= +github.com/pascaldekloe/goe v0.1.0 h1:cBOtyMzM9HTpWjXfbbunk26uA6nG3a8n06Wieeh0MwY= github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= github.com/paypal/gatt v0.0.0-20151011220935-4ae819d591cf/go.mod h1:+AwQL2mK3Pd3S+TUwg0tYQjid0q1txyNUJuuSmz8Kdk= github.com/pelletier/go-toml v1.7.0/go.mod h1:vwGMzjaWMwyfHwgIBhI2YUM4fB6nL6lVAvS1LBMMhTE= github.com/pelletier/go-toml v1.9.4 h1:tjENF6MfZAg8e4ZmZTeWaWiT2vXtsoO6+iuOjFhECwM= github.com/pelletier/go-toml v1.9.4/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= +github.com/peterbourgon/g2s v0.0.0-20140925154142-ec76db4c1ac1 h1:5Dl+ADmsGerAqHwWzyLqkNaUBQ+48DQwfDCaW1gHAQM= +github.com/peterbourgon/g2s v0.0.0-20140925154142-ec76db4c1ac1/go.mod h1:1VcHEd3ro4QMoHfiNl/j7Jkln9+KQuorp0PItHMJYNg= +github.com/philhofer/fwd v1.1.1 h1:GdGcTjf5RNAxwS4QLsiMzJYj5KEvPJD3Abr261yRQXQ= +github.com/philhofer/fwd v1.1.1/go.mod h1:gk3iGcWd9+svBvR0sR+KPcfE+RNWozjowpeBVG3ZVNU= github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY= github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= +github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s= +github.com/prometheus/client_golang v0.8.0/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= +github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= +github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= +github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= +github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/common v0.0.0-20180801064454-c7de2306084e/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro= +github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= +github.com/prometheus/procfs v0.0.0-20180725123919-05ee40e3a273/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= +github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= github.com/rhnvrm/simples3 v0.6.1/go.mod h1:Y+3vYm2V7Y4VijFoJHHTrja6OgPrJ2cBti8dPGkC3sA= github.com/rivo/uniseg v0.1.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= +github.com/rpcxio/libkv v0.5.1-0.20210420120011-1fceaedca8a5 h1:oGficf/KJp1y22zTpjjCRtjtNM9QRjww3fqyQPLgypg= +github.com/rpcxio/libkv v0.5.1-0.20210420120011-1fceaedca8a5/go.mod h1:zHGgtLr3cFhGtbalum0BrMPOjhFZFJXCKiws/25ewls= +github.com/rs/cors v1.8.2 h1:KCooALfAYGs415Cwu5ABvv9n9509fSiG5SQJn/AQo4U= +github.com/rs/cors v1.8.2/go.mod h1:XyqrcTp5zjWr1wsJ8PIRZssZ8b/WMcMf71DJnit4EMU= github.com/rs/xid v1.3.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= github.com/rs/zerolog v1.26.0 h1:ORM4ibhEZeTeQlCojCK2kPz1ogAY4bGs4tD+SaAdGaE= github.com/rs/zerolog v1.26.0/go.mod h1:yBiM87lvSqX8h0Ww4sdzNSkVYZ8dL2xjZJG1lAuGZEo= +github.com/rubyist/circuitbreaker v2.2.1+incompatible h1:KUKd/pV8Geg77+8LNDwdow6rVCAYOp8+kHUyFvL6Mhk= +github.com/rubyist/circuitbreaker v2.2.1+incompatible/go.mod h1:Ycs3JgJADPuzJDwffe12k6BZT8hxVi6lFK+gWYJLN4A= +github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo= +github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= github.com/russross/blackfriday/v2 v2.0.1 h1:lPqVAte+HuHNfhJ/0LC98ESWRz8afy9tM/0RK8m9o+Q= github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= github.com/ryanuber/columnize v2.1.0+incompatible/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= github.com/ryanuber/go-glob v1.0.0/go.mod h1:807d1WSdnB0XRJzKNil9Om6lcp/3a0v4qIHxIXzX/Yc= +github.com/samuel/go-zookeeper v0.0.0-20201211165307-7117e9ea2414 h1:AJNDS0kP60X8wwWFvbLPwDuojxubj9pbfK7pjHw0vKg= +github.com/samuel/go-zookeeper v0.0.0-20201211165307-7117e9ea2414/go.mod h1:gi+0XIa01GRL2eRQVjQkKGqKF3SF9vZR/HnPullcV2E= +github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529 h1:nn5Wsu0esKSJiIVhscUtVbo7ada43DJhG55ua/hjS5I= +github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= +github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= +github.com/serialx/hashring v0.0.0-20200727003509-22c0c7ab6b1b/go.mod h1:/yeG0My1xr/u+HZrFQ1tOQQQQrOawfyMUH13ai5brBc= +github.com/shurcooL/component v0.0.0-20170202220835-f88ec8f54cc4/go.mod h1:XhFIlyj5a1fBNx5aJTbKoIq0mNaPvOagO+HjB3EtxrY= +github.com/shurcooL/events v0.0.0-20181021180414-410e4ca65f48/go.mod h1:5u70Mqkb5O5cxEA8nxTsgrgLehJeAw6Oc4Ab1c/P1HM= +github.com/shurcooL/github_flavored_markdown v0.0.0-20181002035957-2122de532470/go.mod h1:2dOwnU2uBioM+SGy2aZoq1f/Sd1l9OkAeAUvjSyvgU0= +github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk= +github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ= +github.com/shurcooL/gofontwoff v0.0.0-20180329035133-29b52fc0a18d/go.mod h1:05UtEgK5zq39gLST6uB0cf3NEHjETfB4Fgr3Gx5R9Vw= +github.com/shurcooL/gopherjslib v0.0.0-20160914041154-feb6d3990c2c/go.mod h1:8d3azKNyqcHP1GaQE/c6dDgjkgSx2BZ4IoEi4F1reUI= +github.com/shurcooL/highlight_diff v0.0.0-20170515013008-09bb4053de1b/go.mod h1:ZpfEhSmds4ytuByIcDnOLkTHGUI6KNqRNPDLHDk+mUU= +github.com/shurcooL/highlight_go v0.0.0-20181028180052-98c3abbbae20/go.mod h1:UDKB5a1T23gOMUJrI+uSuH0VRDStOiUVSjBTRDVBVag= +github.com/shurcooL/home v0.0.0-20181020052607-80b7ffcb30f9/go.mod h1:+rgNQw2P9ARFAs37qieuu7ohDNQ3gds9msbT2yn85sg= +github.com/shurcooL/htmlg v0.0.0-20170918183704-d01228ac9e50/go.mod h1:zPn1wHpTIePGnXSHpsVPWEktKXHr6+SS6x/IKRb7cpw= +github.com/shurcooL/httperror v0.0.0-20170206035902-86b7830d14cc/go.mod h1:aYMfkZ6DWSJPJ6c4Wwz3QtW22G7mf/PEgaB9k/ik5+Y= +github.com/shurcooL/httpfs v0.0.0-20171119174359-809beceb2371/go.mod h1:ZY1cvUeJuFPAdZ/B6v7RHavJWZn2YPVFQ1OSXhCGOkg= +github.com/shurcooL/httpgzip v0.0.0-20180522190206-b1c53ac65af9/go.mod h1:919LwcH0M7/W4fcZ0/jy0qGght1GIhqyS/EgWGH2j5Q= +github.com/shurcooL/issues v0.0.0-20181008053335-6292fdc1e191/go.mod h1:e2qWDig5bLteJ4fwvDAc2NHzqFEthkqn7aOZAOpj+PQ= +github.com/shurcooL/issuesapp v0.0.0-20180602232740-048589ce2241/go.mod h1:NPpHK2TI7iSaM0buivtFUc9offApnI0Alt/K8hcHy0I= +github.com/shurcooL/notifications v0.0.0-20181007000457-627ab5aea122/go.mod h1:b5uSkrEVM1jQUspwbixRBhaIjIzL2xazXp6kntxYle0= +github.com/shurcooL/octicon v0.0.0-20181028054416-fa4f57f9efb2/go.mod h1:eWdoE5JD4R5UVWDucdOPg1g2fqQRq78IQa9zlOV1vpQ= +github.com/shurcooL/reactions v0.0.0-20181006231557-f2e0b4ca5b82/go.mod h1:TCR1lToEk4d2s07G3XGfz2QrgHXg4RJBvjrOozvoWfk= +github.com/shurcooL/sanitized_anchor_name v0.0.0-20170918181015-86672fcb3f95/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo= github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +github.com/shurcooL/users v0.0.0-20180125191416-49c67e49c537/go.mod h1:QJTqeLYEDaXHZDBsXlPCDqdhQuJkuw4NOtaxYe3xii4= +github.com/shurcooL/webdavfs v0.0.0-20170829043945-18c3829fa133/go.mod h1:hKmq5kWdCj2z2KEozexVbfEZIWiTjhE0+UjmZgPqehw= +github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= +github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= github.com/sirupsen/logrus v1.8.1 h1:dJKuHgqk1NNQlqoA6BTlM1Wf9DOH3NBjQyu0h9+AZZE= github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= +github.com/smallnest/quick v0.0.0-20220103065406-780def6371e6 h1:J8xk0QMMrqfDLqU0m07pYRRiOIlE7I3dNWAp/pAHqfo= +github.com/smallnest/quick v0.0.0-20220103065406-780def6371e6/go.mod h1:h+J5yoLzf3XMKtM9l4vOaUtS4e+si6T3sKDtheJ15wc= +github.com/smallnest/rpcx v1.7.4 h1:u6ADk/Ep8BqtAoJZO7LbniWsP+nqeAtcbaPm2D4eOXg= +github.com/smallnest/rpcx v1.7.4/go.mod h1:TSciUoPlm8MYxnC7ErCz5ZymOFxOTbhN9cRgEI6Degs= +github.com/soheilhy/cmux v0.1.5 h1:jjzc5WVemNEDTLwv9tlmemhC73tI08BNOIGwBOo10Js= +github.com/soheilhy/cmux v0.1.5/go.mod h1:T7TcVDs9LWfQgPlPsdngu6I6QIoyIFZDDC6sNE1GqG0= +github.com/sourcegraph/annotate v0.0.0-20160123013949-f4cad6c6324d/go.mod h1:UdhH50NIW0fCiwBSr0co2m7BnFLdv4fQTgdqdJTHFeE= +github.com/sourcegraph/syntaxhighlight v0.0.0-20170531221838-bd320f5d308e/go.mod h1:HuIsMU8RRBOtsCgI77wP899iHVBQpCmg4ErYMZB+2IA= github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= @@ -183,121 +512,298 @@ github.com/srwiley/rasterx v0.0.0-20200120212402-85cb7272f5e9/go.mod h1:mvWM0+15 github.com/srwiley/rasterx v0.0.0-20210519020934-456a8d69b780 h1:oDMiXaTMyBEuZMU53atpxqYsSB3U1CHkeAu2zr6wTeY= github.com/srwiley/rasterx v0.0.0-20210519020934-456a8d69b780/go.mod h1:mvWM0+15UqyrFKqdRjY6LuAVJR0HOVhJlEgZ5JWtSWU= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.3.0 h1:NGXK3lHquSN08v5vWalVI/L8XU9hdzE/G6xsrze47As= +github.com/stretchr/objx v0.3.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.1 h1:5TQK59W5E3v0r2duFAb7P95B6hEeOyEnHRa8MjYSMTY= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/suapapa/go_eddystone v1.3.1/go.mod h1:bXC11TfJOS+3g3q/Uzd7FKd5g62STQEfeEIhcKe4Qy8= +github.com/tarm/serial v0.0.0-20180830185346-98f6abe2eb07/go.mod h1:kDXzergiv9cbyO7IOYJZWg1U88JhDg3PB6klq9Hg2pA= +github.com/templexxx/cpufeat v0.0.0-20180724012125-cef66df7f161 h1:89CEmDvlq/F7SJEOqkIdNDGJXrQIhuIx9D2DBXjavSU= +github.com/templexxx/cpufeat v0.0.0-20180724012125-cef66df7f161/go.mod h1:wM7WEvslTq+iOEAMDLSzhVuOt5BRZ05WirO+b09GHQU= +github.com/templexxx/xor v0.0.0-20191217153810-f85b25db303b h1:fj5tQ8acgNUr6O8LEplsxDhUIe2573iLkJc+PqnzZTI= +github.com/templexxx/xor v0.0.0-20191217153810-f85b25db303b/go.mod h1:5XA7W9S6mni3h5uvOC75dA3m9CCCaS83lltmc0ukdi4= +github.com/tinylib/msgp v1.1.6 h1:i+SbKraHhnrf9M5MYmvQhFnbLhAXSDWF8WWsuyRdocw= +github.com/tinylib/msgp v1.1.6/go.mod h1:75BAfg2hauQhs3qedfdDZmWAPcFMAvJE5b9rGOMufyw= +github.com/tjfoc/gmsm v1.4.1 h1:aMe1GlZb+0bLjn+cKTPEvvn9oUEBlJitaZiiBwsbgho= +github.com/tjfoc/gmsm v1.4.1/go.mod h1:j4INPkHWMrhJb38G+J6W4Tw0AbuN8Thu3PbdVYhVcTE= +github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= github.com/urfave/cli/v2 v2.3.0 h1:qph92Y649prgesehzOrQjdWyxFOp/QVM+6imKHad91M= github.com/urfave/cli/v2 v2.3.0/go.mod h1:LJmUH05zAU44vOAcrfzZQKsZbVcdbOG8rtL3/XcUArI= +github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= +github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= +github.com/valyala/fastrand v1.1.0 h1:f+5HkLW4rsgzdNoleUOB69hyT9IlD2ZQh9GyDMfb5G8= +github.com/valyala/fastrand v1.1.0/go.mod h1:HWqCzkrkg6QXT8V2EXWvXCoow7vLwOFN002oeRzjapQ= +github.com/viant/assertly v0.4.8/go.mod h1:aGifi++jvCrUaklKEKT0BU95igDNaqkvz+49uaYMPRU= +github.com/viant/toolbox v0.24.0/go.mod h1:OxMCG57V0PXuIP2HNQrtJf2CjqdmbrOx5EkMILuUhzM= +github.com/vmihailenco/msgpack/v5 v5.3.5 h1:5gO0H1iULLWGhs2H5tbAHIZTV8/cYafcFOr9znI5mJU= +github.com/vmihailenco/msgpack/v5 v5.3.5/go.mod h1:7xyJ9e+0+9SaZT0Wt1RGleJXzli6Q/V5KbhBonMG9jc= +github.com/vmihailenco/tagparser/v2 v2.0.0 h1:y09buUbR+b5aycVFQs/g70pqKVZNBmxwAhO7/IwNM9g= +github.com/vmihailenco/tagparser/v2 v2.0.0/go.mod h1:Wri+At7QHww0WTrCBeu4J6bNtoV6mEfg5OIWRZA9qds= github.com/x448/float16 v0.8.4 h1:qLwI1I70+NjRFUR3zs1JPUCgaCXSh3SW62uAKT1mSBM= github.com/x448/float16 v0.8.4/go.mod h1:14CWIYCyZA/cWjXOioeEpHeN/83MdbZDRQHoFcYsOfg= +github.com/xtaci/kcp-go v5.4.20+incompatible h1:TN1uey3Raw0sTz0Fg8GkfM0uH3YwzhnZWQ1bABv5xAg= +github.com/xtaci/kcp-go v5.4.20+incompatible/go.mod h1:bN6vIwHQbfHaHtFpEssmWsN45a+AZwO7eyRCmEIbtvE= +github.com/xtaci/lossyconn v0.0.0-20200209145036-adba10fffc37 h1:EWU6Pktpas0n8lLQwDsRyZfmkPeRbdgPtW609es+/9E= +github.com/xtaci/lossyconn v0.0.0-20200209145036-adba10fffc37/go.mod h1:HpMP7DB2CyokmAh4lp0EQnnWhmycP/TvwBGzvuie+H0= +github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.3.8/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +github.com/yuin/goldmark v1.4.1/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.4.4 h1:zNWRjYUW32G9KirMXYHQHVNFkXvMI7LpgNW2AgYAoIs= github.com/yuin/goldmark v1.4.4/go.mod h1:rmuwmfZ0+bvzB24eSC//bk1R1Zp3hM0OXYv/G2LIilg= go.arsenm.dev/infinitime v0.0.0-20220416112421-b7a50271bece h1:ns/GMc4E8ZUZ9TEXhXgU4M+5sRaOLTFFoBWEJ67p3YM= go.arsenm.dev/infinitime v0.0.0-20220416112421-b7a50271bece/go.mod h1:Prvwx7Y2y8HsNRA1tPptduW9jzuw/JffmocvoHcDbYo= +go.opencensus.io v0.18.0/go.mod h1:vKdFvxhtzZ9onBp9VKHK8z/sRpBMnKAsufL7wlDrCOA= +go.opentelemetry.io/otel v0.19.0/go.mod h1:j9bF567N9EfomkSidSfmMwIwIBuP37AMAIzVW85OxSg= +go.opentelemetry.io/otel v1.6.3 h1:FLOfo8f9JzFVFVyU+MSRJc2HdEAXQgm7pIv2uFKRSZE= +go.opentelemetry.io/otel v1.6.3/go.mod h1:7BgNga5fNlF/iZjG06hM3yofffp0ofKCDwSXx1GC4dI= +go.opentelemetry.io/otel/metric v0.19.0/go.mod h1:8f9fglJPRnXuskQmKpnad31lcLJ2VmNNqIsx/uIwBSc= +go.opentelemetry.io/otel/oteltest v0.19.0/go.mod h1:tI4yxwh8U21v7JD6R3BcA/2+RBoTKFexE/PJ/nSO7IA= +go.opentelemetry.io/otel/trace v0.19.0/go.mod h1:4IXiNextNOpPnRlI4ryK69mn5iC84bjBWZQA5DXz/qg= +go.opentelemetry.io/otel/trace v1.6.3 h1:IqN4L+5b0mPNjdXIiZ90Ni4Bl5BRkDQywePLWemd9bc= +go.opentelemetry.io/otel/trace v1.6.3/go.mod h1:GNJQusJlUgZl9/TQBPKU/Y/ty+0iVB5fjhKeJGZPGFs= +go4.org v0.0.0-20180809161055-417644f6feb5/go.mod h1:MkTOUMDaeVYJUOUsaDXIhWPZYa1yOyC1qaOBpL57BhE= +golang.org/x/build v0.0.0-20190111050920-041ab4dc3f9d/go.mod h1:OWs+y06UdEOHN4y+MfF/py+xQ/tYqIWW03b70/CG9Rw= +golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20181030102418-4d3f4d9ffa16/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190313024323-a1f597ede03a/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20200221231518-2aa609cf4a9d/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20201012173705-84dcc777aaee/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= +golang.org/x/crypto v0.0.0-20220408190544-5352b0902921 h1:iU7T1X1J6yxDr0rda54sWGkHgOp5XJrqm79gcNlC2VM= +golang.org/x/crypto v0.0.0-20220408190544-5352b0902921/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/image v0.0.0-20200430140353-33d19683fad8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= golang.org/x/image v0.0.0-20211028202545-6944b10bf410 h1:hTftEOvwiOq2+O8k2D5/Q7COC7k5Qcrgc2TFURJYnvQ= golang.org/x/image v0.0.0-20211028202545-6944b10bf410/go.mod h1:023OzeP/+EPmXeapQh35lcL3II3LrY8Ic+EFFKVhULM= +golang.org/x/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= +golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.5.1/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro= +golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3 h1:kQgndtyPBW/JIYERgdxfwMYh3AVStj88WQTlNDi2a+o= +golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3/go.mod h1:3p9vT2HGsQu2K1YbXdKPJLVgG5VJdoTa1poYQBtP1AY= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181029044818-c44066c5c816/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181106065722-10aee1819953/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190313220215-9f648a60d977/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20201010224723-4f7140c49acb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1/go.mod h1:9tjilg8BloeKEkVJvy7fQ90B1CfIiPueXVOjqfkSzI8= +golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= +golang.org/x/net v0.0.0-20210726213435-c6fcb2dbf985/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20211209124913-491a49abca63 h1:iocB37TsdFuN6IBRZ+ry36wrkoV51/tl5vOWqkcPGvY= -golang.org/x/net v0.0.0-20211209124913-491a49abca63/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20211015210444-4f30a5c0130f/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20211216030914-fe4d6282115f/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20220407224826-aac1ed45d8e3 h1:EN5+DfgmRMvRUrMGERW2gQl3Vc+Z7ZMnI/xdEpPSf0c= +golang.org/x/net v0.0.0-20220407224826-aac1ed45d8e3/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20181017192945-9dcd33a902f4/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20181203162652-d668ce993890/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/perf v0.0.0-20180704124530-6e6d33e29852/go.mod h1:JLpeXjPJfIyPr5TlbXLkXWLhP8nz10XfvxElABhCtcw= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181029174526-d69651ed3497/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190129075346-302c3dd5f1cc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190316082340-a2f829d7f35f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200728102440-3e129f6d46b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220209214540-3681064d5158 h1:rm+CHSpPEEW2IsXUib1ThaHIjuBVZjxNgSKmBLFfD4c= +golang.org/x/sys v0.0.0-20211019181941-9d821ace8654/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220209214540-3681064d5158/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220408201424-a24fb2fb8a0f h1:8w7RhxzTVgUzw/AH/9mUV5q0vMgy40SQRursCcfmkCw= +golang.org/x/sys v0.0.0-20220408201424-a24fb2fb8a0f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.1-0.20181227161524-e6919f6577db/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= +golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20181030000716-a0a13e073c7b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191216052735-49a3e744a425/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200925191224-5d1fdd8fa346/go.mod h1:z6u4i615ZeAfBE4XtMziQW1fSVJXACjjbWkB/mvPzlU= +golang.org/x/tools v0.0.0-20201022035929-9cf592e881e9/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.6-0.20210726203631-07bc1bf47fb2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.7/go.mod h1:LGqMHiF4EqQNHR1JncWGqT5BVaXmza+X+BDGol+dOxo= +golang.org/x/tools v0.1.8/go.mod h1:nABZi5QlRsZVlzPpHl034qft6wpY4eDcsTt5AaioBiU= +golang.org/x/tools v0.1.10 h1:QjFRCZxdOhBJ/UNgnBZLbNV13DlbnK0quyivTnXJM20= +golang.org/x/tools v0.1.10/go.mod h1:Uh6Zz+xoGYZom868N8YTex3t7RhtHDBrE8Gzo9bV56E= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0= +google.golang.org/api v0.0.0-20181030000543-1d582fd0359e/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0= +google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.2.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/appengine v1.3.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20180831171423-11092d34479b/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20181029155118-b69ba1387ce2/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20181202183823-bd91e49a0898/go.mod h1:7Ep/1NZk928CDR8SjdVbjWNpdIf6nzjE3BTgJDr2Atg= +google.golang.org/genproto v0.0.0-20190306203927-b5d61aea6440/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190404172233-64821d5d2107/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= google.golang.org/grpc v1.14.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw= +google.golang.org/grpc v1.16.0/go.mod h1:0JHn/cJsOMiMfNA9+DeHDlAU7KAAB5GDlYFpa9MZMio= +google.golang.org/grpc v1.17.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.22.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= +google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= +google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= +google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= +google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= +google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= +google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= +google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= +google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= +google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= +google.golang.org/protobuf v1.28.0 h1:w43yiav+6bVFTBQFZX0r7ipe9JQ1QsbMgHwbBziscLw= +google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= +gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/asn1-ber.v1 v1.0.0-20181015200546-f715ec2f112d/go.mod h1:cuepJuh7vyXfUyUwEgHQXw849cJrilpS5NeIjOWESAw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU= gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= +gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= gopkg.in/square/go-jose.v2 v2.3.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +grpc.go4.org v0.0.0-20170609214715-11d0a25b4919/go.mod h1:77eQGdRu53HpSqPFJFmuJdjuHRquDANNeA4x7B8WQ9o= +honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +sourcegraph.com/sourcegraph/go-diff v0.5.0/go.mod h1:kuch7UrkMzY0X+p9CRK03kfuPQ2zzQcaEFbx8wA8rck= +sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4/go.mod h1:ketZ/q3QxT9HOBeFhu6RdvsftgpsbFHBF5Cas6cDKZ0= +stathat.com/c/consistent v1.0.1-0.20190920041245-30ea17e8a481/go.mod h1:QkzMWzcbB+yQBL2AttO6sgsQS/JSTapcDISJalmCDS0= diff --git a/internal/types/types.go b/internal/types/types.go deleted file mode 100644 index 13c8507..0000000 --- a/internal/types/types.go +++ /dev/null @@ -1,150 +0,0 @@ -package types - -import ( - "fmt" - "strconv" -) - -const ( - ReqTypeHeartRate = iota - ReqTypeBattLevel - ReqTypeFwVersion - ReqTypeFwUpgrade - ReqTypeBtAddress - ReqTypeNotify - ReqTypeSetTime - ReqTypeWatchHeartRate - ReqTypeWatchBattLevel - ReqTypeMotion - ReqTypeWatchMotion - ReqTypeStepCount - ReqTypeWatchStepCount - ReqTypeCancel - ReqTypeFS - ReqTypeWeatherUpdate -) - -const ( - UpgradeTypeArchive = iota - UpgradeTypeFiles -) - -const ( - FSTypeWrite = iota - FSTypeRead - FSTypeMove - FSTypeDelete - FSTypeList - FSTypeMkdir -) - -type ReqDataFS struct { - Type int `json:"type"` - Files []string `json:"files"` - Data string `json:"data,omitempty"` -} - -type ReqDataFwUpgrade struct { - Type int - Files []string -} - -type Response struct { - Type int `json:"type"` - Value interface{} `json:"value,omitempty"` - Message string `json:"msg,omitempty"` - ID string `json:"id,omitempty"` - Error bool `json:"error"` -} - -type Request struct { - Type int `json:"type"` - Data interface{} `json:"data,omitempty"` -} - -type ReqDataNotify struct { - Title string - Body string -} - -type DFUProgress struct { - Received int64 `mapstructure:"recvd"` - Total int64 `mapstructure:"total"` - Sent int64 `mapstructure:"sent"` -} - -type FSTransferProgress struct { - Type int `json:"type" mapstructure:"type"` - Total uint32 `json:"total" mapstructure:"total"` - Sent uint32 `json:"sent" mapstructure:"sent"` - Done bool `json:"done" mapstructure:"done"` -} - -type MotionValues struct { - X int16 - Y int16 - Z int16 -} - -type FileInfo struct { - Name string `json:"name"` - Size int64 `json:"size"` - IsDir bool `json:"isDir"` -} - -func (fi FileInfo) String() string { - var isDirChar rune - if fi.IsDir { - isDirChar = 'd' - } else { - isDirChar = '-' - } - - // Get human-readable value for file size - val, unit := bytesHuman(fi.Size) - prec := 0 - // If value is less than 10, set precision to 1 - if val < 10 { - prec = 1 - } - // Convert float to string - valStr := strconv.FormatFloat(val, 'f', prec, 64) - - // Return string formatted like so: - // - 10 kB file - // or: - // d 0 B . - return fmt.Sprintf( - "%c %3s %-2s %s", - isDirChar, - valStr, - unit, - fi.Name, - ) -} - -// bytesHuman returns a human-readable string for -// the amount of bytes inputted. -func bytesHuman(b int64) (float64, string) { - const unit = 1000 - // Set possible units prefixes (PineTime flash is 4MB) - units := [2]rune{'k', 'M'} - // If amount of bytes is less than smallest unit - if b < unit { - // Return unchanged with unit "B" - return float64(b), "B" - } - - div, exp := int64(unit), 0 - // Get decimal values and unit prefix index - for n := b / unit; n >= unit; n /= unit { - div *= unit - exp++ - } - - // Create string for full unit - unitStr := string([]rune{units[exp], 'B'}) - - // Return decimal with unit string - return float64(b) / float64(div), unitStr -} diff --git a/socket.go b/socket.go index 4f029a6..f727891 100644 --- a/socket.go +++ b/socket.go @@ -19,9 +19,8 @@ package main import ( - "bufio" - "encoding/json" - "fmt" + "context" + "errors" "io" "net" "os" @@ -30,12 +29,22 @@ import ( "time" "github.com/google/uuid" - "github.com/mitchellh/mapstructure" "github.com/rs/zerolog/log" + "github.com/smallnest/rpcx/server" + "github.com/vmihailenco/msgpack/v5" "go.arsenm.dev/infinitime" "go.arsenm.dev/infinitime/blefs" - "go.arsenm.dev/itd/internal/types" - "go.arsenm.dev/itd/translit" + "go.arsenm.dev/itd/api" +) + +// This type signifies an unneeded valie. +// A struct{} is used as it takes no space in memory. +type none = struct{} + +var ( + ErrDFUInvalidFile = errors.New("provided file is invalid for given upgrade type") + ErrDFUNotEnoughFiles = errors.New("not enough files provided for given upgrade type") + ErrDFUInvalidUpgType = errors.New("invalid upgrade type") ) type DoneMap map[string]chan struct{} @@ -85,18 +94,36 @@ func startSocket(dev *infinitime.Device) error { log.Warn().Err(err).Msg("Error getting BLE filesystem") } - go func() { - for { - // Accept socket connection - conn, err := ln.Accept() - if err != nil { - log.Error().Err(err).Msg("Error accepting connection") - } + srv := server.NewServer() - // Concurrently handle connection - go handleConnection(conn, dev, fs) - } - }() + itdAPI := &ITD{ + dev: dev, + srv: srv, + } + srv.Register(itdAPI, "") + + fsAPI := &FS{ + dev: dev, + fs: fs, + srv: srv, + } + srv.Register(fsAPI, "") + + go srv.ServeListener("unix", ln) + + /* + go func() { + for { + // Accept socket connection + conn, err := ln.Accept() + if err != nil { + log.Error().Err(err).Msg("Error accepting connection") + } + + // Concurrently handle connection + go handleConnection(conn, dev, fs) + } + }()*/ // Log socket start log.Info().Str("path", k.String("socket.path")).Msg("Started control socket") @@ -104,580 +131,439 @@ func startSocket(dev *infinitime.Device) error { return nil } -func handleConnection(conn net.Conn, dev *infinitime.Device, fs *blefs.FS) { - defer conn.Close() +type ITD struct { + dev *infinitime.Device + srv *server.Server +} - // If an FS update is required (reconnect ocurred) - if updateFS { +func (i *ITD) HeartRate(_ context.Context, _ none, out *uint8) error { + heartRate, err := i.dev.HeartRate() + *out = heartRate + return err +} + +func (i *ITD) WatchHeartRate(ctx context.Context, _ none, out *string) error { + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + heartRateCh, cancel, err := i.dev.WatchHeartRate() + if err != nil { + return err + } + + id := uuid.New().String() + go func() { + done.Create(id) + // For every heart rate value + for heartRate := range heartRateCh { + select { + case <-done[id]: + // Stop notifications if done signal received + cancel() + done.Remove(id) + return + default: + data, err := msgpack.Marshal(heartRate) + if err != nil { + continue + } + + // Send response to connection if no done signal received + i.srv.SendMessage(clientConn, id, "HeartRateSample", nil, data) + } + } + }() + + *out = id + return nil +} + +func (i *ITD) BatteryLevel(_ context.Context, _ none, out *uint8) error { + battLevel, err := i.dev.BatteryLevel() + *out = battLevel + return err +} + +func (i *ITD) WatchBatteryLevel(ctx context.Context, _ none, out *string) error { + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + battLevelCh, cancel, err := i.dev.WatchBatteryLevel() + if err != nil { + return err + } + + id := uuid.New().String() + go func() { + done.Create(id) + // For every heart rate value + for battLevel := range battLevelCh { + select { + case <-done[id]: + // Stop notifications if done signal received + cancel() + done.Remove(id) + return + default: + data, err := msgpack.Marshal(battLevel) + if err != nil { + continue + } + + // Send response to connection if no done signal received + i.srv.SendMessage(clientConn, id, "BatteryLevelSample", nil, data) + } + } + }() + + *out = id + return nil +} + +func (i *ITD) Motion(_ context.Context, _ none, out *infinitime.MotionValues) error { + motionVals, err := i.dev.Motion() + *out = motionVals + return err +} + +func (i *ITD) WatchMotion(ctx context.Context, _ none, out *string) error { + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + motionValsCh, cancel, err := i.dev.WatchMotion() + if err != nil { + return err + } + + id := uuid.New().String() + go func() { + done.Create(id) + // For every heart rate value + for motionVals := range motionValsCh { + select { + case <-done[id]: + // Stop notifications if done signal received + cancel() + done.Remove(id) + return + default: + data, err := msgpack.Marshal(motionVals) + if err != nil { + continue + } + + // Send response to connection if no done signal received + i.srv.SendMessage(clientConn, id, "MotionSample", nil, data) + } + } + }() + + *out = id + return nil +} + +func (i *ITD) StepCount(_ context.Context, _ none, out *uint32) error { + stepCount, err := i.dev.StepCount() + *out = stepCount + return err +} + +func (i *ITD) WatchStepCount(ctx context.Context, _ none, out *string) error { + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + stepCountCh, cancel, err := i.dev.WatchStepCount() + if err != nil { + return err + } + + id := uuid.New().String() + go func() { + done.Create(id) + // For every heart rate value + for stepCount := range stepCountCh { + select { + case <-done[id]: + // Stop notifications if done signal received + cancel() + done.Remove(id) + return + default: + data, err := msgpack.Marshal(stepCount) + if err != nil { + continue + } + + // Send response to connection if no done signal received + i.srv.SendMessage(clientConn, id, "StepCountSample", nil, data) + } + } + }() + + *out = id + return nil +} + +func (i *ITD) Version(_ context.Context, _ none, out *string) error { + version, err := i.dev.Version() + *out = version + return err +} + +func (i *ITD) Address(_ context.Context, _ none, out *string) error { + addr := i.dev.Address() + *out = addr + return nil +} + +func (i *ITD) Notify(_ context.Context, data api.NotifyData, _ *none) error { + return i.dev.Notify(data.Title, data.Body) +} + +func (i *ITD) SetTime(_ context.Context, t time.Time, _ *none) error { + return i.dev.SetTime(t) +} + +func (i *ITD) WeatherUpdate(_ context.Context, _ none, _ *none) error { + sendWeatherCh <- struct{}{} + return nil +} + +func (i *ITD) FirmwareUpgrade(ctx context.Context, reqData api.FwUpgradeData, out *string) error { + i.dev.DFU.Reset() + + switch reqData.Type { + case api.UpgradeTypeArchive: + // If less than one file, return error + if len(reqData.Files) < 1 { + return ErrDFUNotEnoughFiles + } + // If file is not zip archive, return error + if filepath.Ext(reqData.Files[0]) != ".zip" { + return ErrDFUInvalidFile + } + // Load DFU archive + err := i.dev.DFU.LoadArchive(reqData.Files[0]) + if err != nil { + return err + } + case api.UpgradeTypeFiles: + // If less than two files, return error + if len(reqData.Files) < 2 { + return ErrDFUNotEnoughFiles + } + // If first file is not init packet, return error + if filepath.Ext(reqData.Files[0]) != ".dat" { + return ErrDFUInvalidFile + } + // If second file is not firmware image, return error + if filepath.Ext(reqData.Files[1]) != ".bin" { + return ErrDFUInvalidFile + } + // Load individual DFU files + err := i.dev.DFU.LoadFiles(reqData.Files[0], reqData.Files[1]) + if err != nil { + return err + } + default: + return ErrDFUInvalidUpgType + } + + id := uuid.New().String() + *out = id + + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + go func() { + // For every progress event + for event := range i.dev.DFU.Progress() { + data, err := msgpack.Marshal(event) + if err != nil { + continue + } + + i.srv.SendMessage(clientConn, id, "DFUProgress", nil, data) + } + + firmwareUpdating = false + i.srv.SendMessage(clientConn, id, "Done", nil, nil) + }() + + // Set firmwareUpdating + firmwareUpdating = true + + go func() { + // Start DFU + err := i.dev.DFU.Start() + if err != nil { + firmwareUpdating = false + return + } + }() + + return nil +} + +func (i *ITD) Done(_ context.Context, id string, _ *none) error { + done.Done(id) + return nil +} + +type FS struct { + dev *infinitime.Device + fs *blefs.FS + srv *server.Server +} + +func (fs *FS) Remove(_ context.Context, paths []string, _ *none) error { + fs.updateFS() + for _, path := range paths { + err := fs.fs.Remove(path) + if err != nil { + return err + } + } + return nil +} + +func (fs *FS) Rename(_ context.Context, paths [2]string, _ *none) error { + fs.updateFS() + return fs.fs.Rename(paths[0], paths[1]) +} + +func (fs *FS) Mkdir(_ context.Context, paths []string, _ *none) error { + fs.updateFS() + for _, path := range paths { + err := fs.fs.Mkdir(path) + if err != nil { + return err + } + } + return nil +} + +func (fs *FS) ReadDir(_ context.Context, dir string, out *[]api.FileInfo) error { + fs.updateFS() + + entries, err := fs.fs.ReadDir(dir) + if err != nil { + return err + } + var fileInfo []api.FileInfo + for _, entry := range entries { + info, err := entry.Info() + if err != nil { + return err + } + fileInfo = append(fileInfo, api.FileInfo{ + Name: info.Name(), + Size: info.Size(), + IsDir: info.IsDir(), + }) + } + + *out = fileInfo + return nil +} + +func (fs *FS) Upload(ctx context.Context, paths [2]string, out *string) error { + fs.updateFS() + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + localFile, err := os.Open(paths[1]) + if err != nil { + return err + } + + localInfo, err := localFile.Stat() + if err != nil { + return err + } + + remoteFile, err := fs.fs.Create(paths[0], uint32(localInfo.Size())) + if err != nil { + return err + } + + id := uuid.New().String() + *out = id + + go func() { + // For every progress event + for sent := range remoteFile.Progress() { + data, err := msgpack.Marshal(api.FSTransferProgress{ + Total: remoteFile.Size(), + Sent: sent, + }) + if err != nil { + continue + } + + fs.srv.SendMessage(clientConn, id, "FSProgress", nil, data) + } + + fs.srv.SendMessage(clientConn, id, "Done", nil, nil) + }() + + go func() { + io.Copy(remoteFile, localFile) + localFile.Close() + remoteFile.Close() + }() + + return nil +} + +func (fs *FS) Download(ctx context.Context, paths [2]string, out *string) error { + fs.updateFS() + clientConn := ctx.Value(server.RemoteConnContextKey).(net.Conn) + + localFile, err := os.Create(paths[0]) + if err != nil { + return err + } + + remoteFile, err := fs.fs.Open(paths[1]) + if err != nil { + return err + } + + id := uuid.New().String() + *out = id + + go func() { + // For every progress event + for rcvd := range remoteFile.Progress() { + data, err := msgpack.Marshal(api.FSTransferProgress{ + Total: remoteFile.Size(), + Sent: rcvd, + }) + if err != nil { + continue + } + + fs.srv.SendMessage(clientConn, id, "FSProgress", nil, data) + } + + fs.srv.SendMessage(clientConn, id, "Done", nil, nil) + localFile.Close() + remoteFile.Close() + }() + + go io.Copy(localFile, remoteFile) + + return nil +} + +func (fs *FS) updateFS() { + if fs.fs == nil || updateFS { // Get new FS - newFS, err := dev.FS() + newFS, err := fs.dev.FS() if err != nil { log.Warn().Err(err).Msg("Error updating BLE filesystem") } else { // Set FS pointer to new FS - *fs = *newFS + fs.fs = newFS // Reset updateFS updateFS = false } } - - // Create new scanner on connection - scanner := bufio.NewScanner(conn) - for scanner.Scan() { - var req types.Request - // Decode scanned message into types.Request - err := json.Unmarshal(scanner.Bytes(), &req) - if err != nil { - connErr(conn, req.Type, err, "Error decoding JSON input") - continue - } - - // If firmware is updating, return error - if firmwareUpdating { - connErr(conn, req.Type, nil, "Firmware update in progress") - return - } - - switch req.Type { - case types.ReqTypeHeartRate: - // Get heart rate from watch - heartRate, err := dev.HeartRate() - if err != nil { - connErr(conn, req.Type, err, "Error getting heart rate") - break - } - // Encode heart rate to connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: heartRate, - }) - case types.ReqTypeWatchHeartRate: - heartRateCh, cancel, err := dev.WatchHeartRate() - if err != nil { - connErr(conn, req.Type, err, "Error getting heart rate channel") - break - } - reqID := uuid.New().String() - go func() { - done.Create(reqID) - // For every heart rate value - for heartRate := range heartRateCh { - select { - case <-done[reqID]: - // Stop notifications if done signal received - cancel() - done.Remove(reqID) - return - default: - // Encode response to connection if no done signal received - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - ID: reqID, - Value: heartRate, - }) - } - } - }() - case types.ReqTypeBattLevel: - // Get battery level from watch - battLevel, err := dev.BatteryLevel() - if err != nil { - connErr(conn, req.Type, err, "Error getting battery level") - break - } - // Encode battery level to connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: battLevel, - }) - case types.ReqTypeWatchBattLevel: - battLevelCh, cancel, err := dev.WatchBatteryLevel() - if err != nil { - connErr(conn, req.Type, err, "Error getting battery level channel") - break - } - reqID := uuid.New().String() - go func() { - done.Create(reqID) - // For every battery level value - for battLevel := range battLevelCh { - select { - case <-done[reqID]: - // Stop notifications if done signal received - cancel() - done.Remove(reqID) - return - default: - // Encode response to connection if no done signal received - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - ID: reqID, - Value: battLevel, - }) - } - } - }() - case types.ReqTypeMotion: - // Get battery level from watch - motionVals, err := dev.Motion() - if err != nil { - connErr(conn, req.Type, err, "Error getting motion values") - break - } - // Encode battery level to connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: motionVals, - }) - case types.ReqTypeWatchMotion: - motionValCh, cancel, err := dev.WatchMotion() - if err != nil { - connErr(conn, req.Type, err, "Error getting heart rate channel") - break - } - reqID := uuid.New().String() - go func() { - done.Create(reqID) - // For every motion event - for motionVals := range motionValCh { - select { - case <-done[reqID]: - // Stop notifications if done signal received - cancel() - done.Remove(reqID) - - return - default: - // Encode response to connection if no done signal received - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - ID: reqID, - Value: motionVals, - }) - } - } - }() - case types.ReqTypeStepCount: - // Get battery level from watch - stepCount, err := dev.StepCount() - if err != nil { - connErr(conn, req.Type, err, "Error getting step count") - break - } - // Encode battery level to connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: stepCount, - }) - case types.ReqTypeWatchStepCount: - stepCountCh, cancel, err := dev.WatchStepCount() - if err != nil { - connErr(conn, req.Type, err, "Error getting heart rate channel") - break - } - reqID := uuid.New().String() - go func() { - done.Create(reqID) - // For every step count value - for stepCount := range stepCountCh { - select { - case <-done[reqID]: - // Stop notifications if done signal received - cancel() - done.Remove(reqID) - return - default: - // Encode response to connection if no done signal received - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - ID: reqID, - Value: stepCount, - }) - } - } - }() - case types.ReqTypeFwVersion: - // Get firmware version from watch - version, err := dev.Version() - if err != nil { - connErr(conn, req.Type, err, "Error getting firmware version") - break - } - // Encode version to connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: version, - }) - case types.ReqTypeBtAddress: - // Encode bluetooth address to connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: dev.Address(), - }) - case types.ReqTypeNotify: - // If no data, return error - if req.Data == nil { - connErr(conn, req.Type, nil, "Data required for notify request") - break - } - var reqData types.ReqDataNotify - // Decode data map to notify request data - err = mapstructure.Decode(req.Data, &reqData) - if err != nil { - connErr(conn, req.Type, err, "Error decoding request data") - break - } - maps := k.Strings("notifs.translit.use") - translit.Transliterators["custom"] = translit.Map(k.Strings("notifs.translit.custom")) - title := translit.Transliterate(reqData.Title, maps...) - body := translit.Transliterate(reqData.Body, maps...) - // Send notification to watch - err = dev.Notify(title, body) - if err != nil { - connErr(conn, req.Type, err, "Error sending notification") - break - } - // Encode empty types.Response to connection - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - case types.ReqTypeSetTime: - // If no data, return error - if req.Data == nil { - connErr(conn, req.Type, nil, "Data required for settime request") - break - } - // Get string from data or return error - reqTimeStr, ok := req.Data.(string) - if !ok { - connErr(conn, req.Type, nil, "Data for settime request must be RFC3339 formatted time string") - break - } - - var reqTime time.Time - if reqTimeStr == "now" { - reqTime = time.Now() - } else { - // Parse time as RFC3339/ISO8601 - reqTime, err = time.Parse(time.RFC3339, reqTimeStr) - if err != nil { - connErr(conn, req.Type, err, "Invalid time format. Time string must be formatted as ISO8601 or the word `now`") - break - } - } - // Set time on watch - err = dev.SetTime(reqTime) - if err != nil { - connErr(conn, req.Type, err, "Error setting device time") - break - } - // Encode empty types.Response to connection - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - case types.ReqTypeFwUpgrade: - // If no data, return error - if req.Data == nil { - connErr(conn, req.Type, nil, "Data required for firmware upgrade request") - break - } - var reqData types.ReqDataFwUpgrade - // Decode data map to firmware upgrade request data - err = mapstructure.Decode(req.Data, &reqData) - if err != nil { - connErr(conn, req.Type, err, "Error decoding request data") - break - } - // Reset DFU to prepare for next update - dev.DFU.Reset() - switch reqData.Type { - case types.UpgradeTypeArchive: - // If less than one file, return error - if len(reqData.Files) < 1 { - connErr(conn, req.Type, nil, "Archive upgrade requires one file with .zip extension") - break - } - // If file is not zip archive, return error - if filepath.Ext(reqData.Files[0]) != ".zip" { - connErr(conn, req.Type, nil, "Archive upgrade file must be a zip archive") - break - } - // Load DFU archive - err := dev.DFU.LoadArchive(reqData.Files[0]) - if err != nil { - connErr(conn, req.Type, err, "Error loading archive file") - break - } - case types.UpgradeTypeFiles: - // If less than two files, return error - if len(reqData.Files) < 2 { - connErr(conn, req.Type, nil, "Files upgrade requires two files. First with .dat and second with .bin extension.") - break - } - // If first file is not init packet, return error - if filepath.Ext(reqData.Files[0]) != ".dat" { - connErr(conn, req.Type, nil, "First file must be a .dat file") - break - } - // If second file is not firmware image, return error - if filepath.Ext(reqData.Files[1]) != ".bin" { - connErr(conn, req.Type, nil, "Second file must be a .bin file") - break - } - // Load individual DFU files - err := dev.DFU.LoadFiles(reqData.Files[0], reqData.Files[1]) - if err != nil { - connErr(conn, req.Type, err, "Error loading firmware files") - break - } - } - - go func() { - // Get progress - progress := dev.DFU.Progress() - // For every progress event - for event := range progress { - // Encode event on connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: event, - }) - } - firmwareUpdating = false - }() - - // Set firmwareUpdating - firmwareUpdating = true - // Start DFU - err = dev.DFU.Start() - if err != nil { - connErr(conn, req.Type, err, "Error performing upgrade") - firmwareUpdating = false - break - } - firmwareUpdating = false - case types.ReqTypeFS: - if fs == nil { - connErr(conn, req.Type, nil, "BLE filesystem is not available") - break - } - - // If no data, return error - if req.Data == nil { - connErr(conn, req.Type, nil, "Data required for filesystem operations") - break - } - - var reqData types.ReqDataFS - // Decode data map to firmware upgrade request data - err = mapstructure.Decode(req.Data, &reqData) - if err != nil { - connErr(conn, req.Type, err, "Error decoding request data") - break - } - - // Clean input filepaths - reqData.Files = cleanPaths(reqData.Files) - - switch reqData.Type { - case types.FSTypeDelete: - if len(reqData.Files) == 0 { - connErr(conn, req.Type, nil, "Remove FS command requires at least one file") - break - } - for _, file := range reqData.Files { - err := fs.Remove(file) - if err != nil { - connErr(conn, req.Type, err, "Error removing file") - break - } - } - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - case types.FSTypeMove: - if len(reqData.Files) != 2 { - connErr(conn, req.Type, nil, "Move FS command requires an old path and new path in the files list") - break - } - err := fs.Rename(reqData.Files[0], reqData.Files[1]) - if err != nil { - connErr(conn, req.Type, err, "Error moving file") - break - } - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - case types.FSTypeMkdir: - if len(reqData.Files) == 0 { - connErr(conn, req.Type, nil, "Mkdir FS command requires at least one file") - break - } - for _, file := range reqData.Files { - err := fs.Mkdir(file) - if err != nil { - connErr(conn, req.Type, err, "Error creating directory") - break - } - } - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - case types.FSTypeList: - if len(reqData.Files) != 1 { - connErr(conn, req.Type, nil, "List FS command requires a path to list in the files list") - break - } - entries, err := fs.ReadDir(reqData.Files[0]) - if err != nil { - connErr(conn, req.Type, err, "Error reading directory") - break - } - var out []types.FileInfo - for _, entry := range entries { - info, err := entry.Info() - if err != nil { - connErr(conn, req.Type, err, "Error getting file info") - break - } - out = append(out, types.FileInfo{ - Name: info.Name(), - Size: info.Size(), - IsDir: info.IsDir(), - }) - } - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: out, - }) - case types.FSTypeWrite: - if len(reqData.Files) != 2 { - connErr(conn, req.Type, nil, "Write FS command requires a path to the file to write") - break - } - - localFile, err := os.Open(reqData.Files[1]) - if err != nil { - connErr(conn, req.Type, err, "Error opening local file") - break - } - defer localFile.Close() - - localInfo, err := localFile.Stat() - if err != nil { - connErr(conn, req.Type, err, "Error getting local file information") - break - } - - remoteFile, err := fs.Create(reqData.Files[0], uint32(localInfo.Size())) - if err != nil { - connErr(conn, req.Type, err, "Error creating remote file") - break - } - defer remoteFile.Close() - - go func() { - // For every progress event - for sent := range remoteFile.Progress() { - // Encode event on connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: types.FSTransferProgress{ - Type: types.FSTypeWrite, - Total: remoteFile.Size(), - Sent: sent, - }, - }) - } - }() - - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - - io.Copy(remoteFile, localFile) - - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: types.FSTransferProgress{ - Type: types.FSTypeWrite, - Total: remoteFile.Size(), - Sent: remoteFile.Size(), - Done: true, - }, - }) - case types.FSTypeRead: - if len(reqData.Files) != 2 { - connErr(conn, req.Type, nil, "Read FS command requires a path to the file to read") - break - } - localFile, err := os.Create(reqData.Files[0]) - if err != nil { - connErr(conn, req.Type, err, "Error creating local file") - break - } - defer localFile.Close() - - remoteFile, err := fs.Open(reqData.Files[1]) - if err != nil { - connErr(conn, req.Type, err, "Error opening remote file") - break - } - defer remoteFile.Close() - - go func() { - // For every progress event - for rcvd := range remoteFile.Progress() { - // Encode event on connection - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: types.FSTransferProgress{ - Type: types.FSTypeRead, - Total: remoteFile.Size(), - Sent: rcvd, - }, - }) - } - }() - - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - - io.Copy(localFile, remoteFile) - - json.NewEncoder(conn).Encode(types.Response{ - Type: req.Type, - Value: types.FSTransferProgress{ - Type: types.FSTypeRead, - Total: remoteFile.Size(), - Sent: remoteFile.Size(), - Done: true, - }, - }) - } - case types.ReqTypeWeatherUpdate: - // Send weather update signal - sendWeatherCh <- struct{}{} - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - case types.ReqTypeCancel: - if req.Data == nil { - connErr(conn, req.Type, nil, "No data provided. Cancel request requires request ID string as data.") - continue - } - reqID, ok := req.Data.(string) - if !ok { - connErr(conn, req.Type, nil, "Invalid data. Cancel request required request ID string as data.") - } - // Stop notifications - done.Done(reqID) - json.NewEncoder(conn).Encode(types.Response{Type: req.Type}) - default: - connErr(conn, req.Type, nil, fmt.Sprintf("Unknown request type %d", req.Type)) - } - } -} - -func connErr(conn net.Conn, resType int, err error, msg string) { - var res types.Response - // If error exists, add to types.Response, otherwise don't - if err != nil { - log.Error().Err(err).Msg(msg) - res = types.Response{Message: fmt.Sprintf("%s: %s", msg, err)} - } else { - log.Error().Msg(msg) - res = types.Response{Message: msg, Type: resType} - } - res.Error = true - - // Encode error to connection - json.NewEncoder(conn).Encode(res) } // cleanPaths runs strings.TrimSpace and filepath.Clean