From a4e70ffd31d4e0e7492b8137e33c361e1b0ebdb3 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 11:55:58 +0200 Subject: [PATCH 01/27] docs: Added more documentation to functions and objects in the project --- cmd/collector/main.go | 12 +++++-- cmd/proxy/server/server.go | 32 +++++++++++++++++-- internal/certificate/provider.go | 13 ++++++++ internal/packet/find.go | 4 +++ .../{memserializer.go => mem_serializer.go} | 0 internal/proxy/server.go | 22 +++++++++++++ internal/ratelimit/limiter.go | 15 ++++++++- internal/redirect/handler.go | 14 ++++++-- internal/visualize/server.go | 17 ++++++++-- internal/worker/collector.go | 17 ++++++++++ 10 files changed, 136 insertions(+), 10 deletions(-) rename internal/packet/{memserializer.go => mem_serializer.go} (100%) diff --git a/cmd/collector/main.go b/cmd/collector/main.go index 7876c05..33e1acf 100644 --- a/cmd/collector/main.go +++ b/cmd/collector/main.go @@ -2,6 +2,7 @@ package main import ( "context" + "flag" "log" "time" "waffle/internal/packet" @@ -9,9 +10,16 @@ import ( "waffle/internal/worker" ) -const networkInterfaceDescription = "Intel(R) I211 Gigabit Network Connection" +const ( + defaultNetworkInterfaceDescription = "Intel(R) I211 Gigabit Network Connection" +) func main() { + var ( + networkInterface string + ) + flag.StringVar(&networkInterface, "i", defaultNetworkInterfaceDescription, "Identification of the interface") + ctx := context.Background() log.Println("starting collector") @@ -26,7 +34,7 @@ func main() { collector := worker.NewCollector( cfg, - packet.NewWindowsNetworkInterfaceProvider(networkInterfaceDescription), + packet.NewWindowsNetworkInterfaceProvider(networkInterface), inMemoryPacketSerializer) if err := collector.Run(ctx); err != nil { diff --git a/cmd/proxy/server/server.go b/cmd/proxy/server/server.go index 81e023f..e1ae2ec 100644 --- a/cmd/proxy/server/server.go +++ b/cmd/proxy/server/server.go @@ -19,6 +19,28 @@ import ( "waffle/internal/waf/guard" ) +// Run initializes and starts the Waffle Proxy server with the provided context, configuration, and embedded certificates. +// +// It first sets up signal handling to allow graceful shutdown on receiving an interrupt signal. +// +// The function loads environment-specific configurations, then parses the provided YAML configuration +// to initialize a DNS provider for managing domain names. +// +// Next, it sets up the certificate provider using locally embedded certificates, +// loading custom CA certificates, certificate PEM blocks, and key PEM blocks. +// +// A defense coordinator is initialized to handle security measures like XSS protection, +// along with an in-memory rate limiter to control the number of requests allowed per a given time window (5 minutes). +// +// Additionally, a server for visualizing traffic is set up on port :8081. +// +// The WAF (Web Application Firewall) handler is constructed using a redirect handler, the defender (for security), +// the rate limiter, and a visualizer from the visualization server. +// +// Finally, the main proxy server is started on port :8080, with the guard handler and certificate provider. +// If the proxy server fails to start, the function logs a fatal error. +// +// The function returns nil upon normal completion. func Run(ctx context.Context, yamlConfigBytes []byte, certificates embed.FS) error { _, cancel := signal.NotifyContext(ctx, os.Interrupt) defer cancel() @@ -65,20 +87,24 @@ func Run(ctx context.Context, yamlConfigBytes []byte, certificates embed.FS) err return nil } +// loadLocalCustomCACerts reads the local custom CA certificates from the embedded file system. +// It reads the CA certificate file (ca.crt) located in the ".cert" directory and returns it as a slice of byte slices. +// This CA certificate is used for establishing trust during TLS/SSL handshakes. func loadLocalCustomCACerts(certificates embed.FS) [][]byte { certBytes, _ := certificates.ReadFile(".cert/ca.crt") - return [][]byte{certBytes} } +// loadLocalCertPEMBlock reads the local server certificate (server.crt) from the embedded file system. +// It returns the certificate as a byte slice, which is later used to serve the server's public certificate in TLS/SSL connections. func loadLocalCertPEMBlock(certificates embed.FS) []byte { certBytes, _ := certificates.ReadFile(".cert/server.crt") - return certBytes } +// loadLocalKeyPEMBlock reads the private key (server.key) from the embedded file system. +// It returns the private key as a byte slice, which is paired with the server certificate during TLS/SSL handshakes. func loadLocalKeyPEMBlock(certificates embed.FS) []byte { certBytes, _ := certificates.ReadFile(".cert/server.key") - return certBytes } diff --git a/internal/certificate/provider.go b/internal/certificate/provider.go index eacd53d..c0eaefc 100644 --- a/internal/certificate/provider.go +++ b/internal/certificate/provider.go @@ -13,12 +13,18 @@ type Provider interface { GetCACertificatesPool() (*x509.CertPool, error) } +// LocalCertificatesProvider is a structure that holds custom CA certificates, +// the server certificate PEM block, and the private key PEM block. +// It provides methods for retrieving TLS certificates and a CA certificate pool for establishing secure connections. type LocalCertificatesProvider struct { customCaCerts [][]byte certPEMBlock []byte keyPEMBlock []byte } +// NewLocalCertificatesProvider initializes a new instance of LocalCertificatesProvider. +// It takes custom CA certificates, a certificate PEM block, and a key PEM block as input, +// and returns a LocalCertificatesProvider that can provide certificates for secure connections. func NewLocalCertificatesProvider(caCerts [][]byte, certPEMBlock, keyPEMBlock []byte) *LocalCertificatesProvider { return &LocalCertificatesProvider{ customCaCerts: caCerts, @@ -27,6 +33,9 @@ func NewLocalCertificatesProvider(caCerts [][]byte, certPEMBlock, keyPEMBlock [] } } +// GetTLSCertificate returns a TLS certificate created from the certificate PEM block and key PEM block. +// This is used to provide the server's certificate for TLS/SSL handshakes. +// If an error occurs while loading the key pair, it returns an error. func (l *LocalCertificatesProvider) GetTLSCertificate() (*tls.Certificate, error) { cert, err := tls.X509KeyPair(l.certPEMBlock, l.keyPEMBlock) if err != nil { @@ -36,6 +45,10 @@ func (l *LocalCertificatesProvider) GetTLSCertificate() (*tls.Certificate, error return &cert, nil } +// GetCACertificatesPool returns a pool of CA certificates, including both system CA certificates and +// any custom CA certificates provided to the LocalCertificatesProvider. +// If the system CA certificate pool cannot be loaded, it returns an error. +// Custom CA certificates are appended to the pool to support specific trusted CAs. func (l *LocalCertificatesProvider) GetCACertificatesPool() (*x509.CertPool, error) { caCertPool, err := x509.SystemCertPool() if err != nil { diff --git a/internal/packet/find.go b/internal/packet/find.go index bb86df8..129b7bd 100644 --- a/internal/packet/find.go +++ b/internal/packet/find.go @@ -11,6 +11,7 @@ import ( ) var ( + // ErrNetworkInterfaceNotFound is an error which indicates that network interface was not found ErrNetworkInterfaceNotFound = errors.New("network interface not found") ) @@ -28,6 +29,9 @@ func NewWindowsNetworkInterfaceProvider(interfaceDescription string) *WindowsNet return &WindowsNetworkInterfaceProvider{interfaceDescription: interfaceDescription} } +// GetNetworkInterface retrieves all available interfaces, verifies if interface's description matches description +// in interfaceDescription field of WindowsNetworkInterfaceProvider struct, and returns an interfaces which +// suits that condition. func (w *WindowsNetworkInterfaceProvider) GetNetworkInterface() (*pcap.Interface, error) { interfaces, err := pcap.FindAllDevs() if err != nil { diff --git a/internal/packet/memserializer.go b/internal/packet/mem_serializer.go similarity index 100% rename from internal/packet/memserializer.go rename to internal/packet/mem_serializer.go diff --git a/internal/proxy/server.go b/internal/proxy/server.go index 80af86a..b77f290 100644 --- a/internal/proxy/server.go +++ b/internal/proxy/server.go @@ -61,12 +61,18 @@ var ( alpnProto = "acme-tls/1" ) +// Server represents an HTTP server that listens on a specific address. +// It uses a certificate provider to manage TLS certificates and an HTTP handler to process incoming requests. type Server struct { addr string certificateProvider certificate.Provider handler http.Handler } +// NewServer initializes a new Server instance with the given address, certificate provider, and HTTP handler. +// The address defines where the server will listen for incoming connections. +// The certificateProvider is used to manage the TLS certificates for secure connections. +// The handler (usually a redirect handler) will process HTTP requests that are received by the server. func NewServer( addr string, certificateProvider certificate.Provider, @@ -79,6 +85,22 @@ func NewServer( } } +// Start initializes and starts the server with TLS configuration using the provided certificate provider. +// It first retrieves the CA certificates pool and server TLS certificate from the certificate provider. +// Then, it configures the server to use TLS 1.3, sets the allowed cipher suites, and prepares the server for +// mutual TLS authentication (client certificates are verified if provided). +// +// A TCP listener is created to listen for incoming TLS connections on the specified address. +// +// The HTTP server is configured with various timeouts (read, write, idle), a maximum header size, and a custom error logger. +// The server handles incoming requests using the provided handler. +// +// A graceful shutdown mechanism is implemented: when an interrupt signal is received (e.g., Ctrl+C), +// the server begins shutting down by closing the active listener and processing any outstanding requests. +// +// The function waits for all idle connections to be closed before returning. +// +// If there is any error while starting or serving the server, the error is returned. func (s *Server) Start() error { caCertPool, err := s.certificateProvider.GetCACertificatesPool() if err != nil { diff --git a/internal/ratelimit/limiter.go b/internal/ratelimit/limiter.go index 13e7330..97c3efc 100644 --- a/internal/ratelimit/limiter.go +++ b/internal/ratelimit/limiter.go @@ -14,15 +14,22 @@ var ( defaultPurgeDuration = time.Minute * 5 ) +// Limiter is an interface that defines rate-limiting functionality for network addresses. +// It includes methods to retrieve the current rate for an IP address and to set a rate limit for an IP address until a specific time. type Limiter interface { GetRate(ctx context.Context, address net.IP) *Rate SetRate(ctx context.Context, address net.IP, limitedUntil time.Time) string } +// InMemoryLimiter is a concrete implementation of the Limiter interface. +// It stores rate limit data in memory using a cache, with each IP address associated with its corresponding rate information. type InMemoryLimiter struct { cache *cache.Cache[stringIpAddress, Rate] } +// NewInMemoryLimiter creates and returns a new instance of InMemoryLimiter. +// It accepts an expirationTime, which defines how long rate limit data will be cached. +// The cache will automatically purge expired data at twice the expiration time or at a default purge duration, whichever is greater. func NewInMemoryLimiter(expirationTime time.Duration) *InMemoryLimiter { purgeDuration := expirationTime * 2 if purgeDuration < defaultPurgeDuration { @@ -36,14 +43,19 @@ func NewInMemoryLimiter(expirationTime time.Duration) *InMemoryLimiter { } } +// Ensures that InMemoryLimiter implements the Limiter interface. var _ Limiter = (*InMemoryLimiter)(nil) +// GetRate retrieves the rate limit information for the given IP address from the in-memory cache. +// If no rate limit is found for the IP address, it returns nil. func (i *InMemoryLimiter) GetRate(_ context.Context, address net.IP) *Rate { rate, _ := i.cache.Get(stringIpAddress(address.String())) - return rate } +// SetRate sets the rate limit for the given IP address until the specified limitedUntil time. +// If a rate limit already exists for the IP address but the existing limit expires later than the provided time, +// the limit is not updated. The method returns a UUID associated with the new rate limit if it is set, or an empty string otherwise. func (i *InMemoryLimiter) SetRate(_ context.Context, address net.IP, limitedUntil time.Time) string { addrString := address.String() @@ -63,4 +75,5 @@ func (i *InMemoryLimiter) SetRate(_ context.Context, address net.IP, limitedUnti return id } +// stringIpAddress is a type alias for string, used to represent IP addresses in the cache. type stringIpAddress string diff --git a/internal/redirect/handler.go b/internal/redirect/handler.go index 75857b2..30aabea 100644 --- a/internal/redirect/handler.go +++ b/internal/redirect/handler.go @@ -6,22 +6,33 @@ import ( "net/http/httputil" "strings" "time" - + "waffle/internal/domain" ) +// Handler is a struct that implements the http.Handler interface. +// It is responsible for processing incoming HTTP requests and forwarding them to the appropriate backend server +// based on the DNS resolution provided by the NameSystemProvider. type Handler struct { dns domain.NameSystemProvider } +// NewHandler creates and returns a new instance of Handler. +// It takes a NameSystemProvider as a parameter, which is used to resolve hostnames to addresses. func NewHandler(dns domain.NameSystemProvider) *Handler { return &Handler{ dns: dns, } } +// Ensures that Handler implements the http.Handler interface. var _ http.Handler = (*Handler)(nil) +// ServeHTTP processes an incoming HTTP request. +// It resolves the target address using the DNS provider based on the request's host. +// If an error occurs during resolution, it responds with a 502 Bad Gateway status. +// Otherwise, it sets up a reverse proxy to the resolved address and forwards the request, +// adjusting the request URL and headers as needed to maintain the correct context for the proxying. func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) { addr, err := h.dns.GetAddress(r.Host) if err != nil { @@ -35,7 +46,6 @@ func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) { r.URL.Scheme = addr.Scheme r.Header.Set("X-Forwarded-Host", r.Header.Get("Host")) r.Host = addr.Host - //trim reverseProxyRoutePrefix path := r.URL.Path r.URL.Path = strings.TrimLeft(path, addr.Path) fmt.Printf("[ TinyRP ] Redirecting request to %s at %s\n", r.URL, time.Now().UTC()) diff --git a/internal/visualize/server.go b/internal/visualize/server.go index e937a7f..389a233 100644 --- a/internal/visualize/server.go +++ b/internal/visualize/server.go @@ -2,16 +2,22 @@ package visualize import ( "context" + "log" "net/http" "nhooyr.io/websocket" ) +// Server represents a web server that listens on a specific port and includes a visualizer component. +// The visualizer is used to analyze or display certain data related to the server's functionality. type Server struct { port string visualizer *Visualizer } +// NewServer initializes and returns a new instance of the Server struct. +// It takes a port as input to define where the server will listen for incoming connections. +// A new Visualizer is also created and attached to the server during initialization. func NewServer(port string) *Server { return &Server{ port: port, @@ -19,19 +25,26 @@ func NewServer(port string) *Server { } } +// GetVisualizer returns the Visualizer instance associated with the server. +// This allows other components to access and use the visualizer for data visualization. func (s *Server) GetVisualizer() *Visualizer { return s.visualizer } +// Start begins the server's operation, listening on the specified port. +// It defines an HTTP handler that attempts to upgrade incoming HTTP connections to WebSocket connections. +// If a connection upgrade fails, the error is silently handled. +// The server listens for incoming requests and serves them using the handler. +// If the server fails to start, an error is returned and handled. func (s *Server) Start(ctx context.Context) { handlerFunc := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { _, err := websocket.Accept(w, r, nil) if err != nil { - + log.Fatalln("Accept error:", err) } }) if err := http.ListenAndServe(s.port, handlerFunc); err != nil { - + log.Fatalln("Error while listening and serving visualizer:", err) } } diff --git a/internal/worker/collector.go b/internal/worker/collector.go index 2c3b749..ed5efd1 100644 --- a/internal/worker/collector.go +++ b/internal/worker/collector.go @@ -47,6 +47,23 @@ func NewCollector( } } +// Run starts the packet capturing process for the Collector. +// It retrieves the network interface using the netInterfaceProvider. +// Then, it opens a live packet capture session on the interface using pcap, with a +// specified snapshot length of 1600 bytes, setting the device to promiscuous mode +// and waiting indefinitely for packets to arrive. +// +// If a BPF is provided in the configuration (c.cfg.BPF), +// it applies the filter to capture only relevant packets. +// +// The packets are read using a gopacket PacketSource and passed through a channel +// to be serialized by the Serializer in a separate goroutine. +// +// The function listens for packets in an infinite loop, and if the context is canceled +// (signaling termination), it gracefully exits. +// +// Deferred actions ensure that the packet channel is closed and a log message is generated +// when the collector is closed. func (c *Collector) Run(ctx context.Context) error { netInterface, err := c.netInterfaceProvider.GetNetworkInterface() if err != nil { From bb9b964fa0d2b502d2440e2a116cb403a7f8eac7 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:21:02 +0200 Subject: [PATCH 02/27] fix: Repaired dockerfile and added commands in Makefile to build/push image with specified IMAGE_NAME variable --- build/Dockerfile | 7 ++++--- makefile | 12 +++++++++++- 2 files changed, 15 insertions(+), 4 deletions(-) diff --git a/build/Dockerfile b/build/Dockerfile index 7d855c8..c52c30e 100644 --- a/build/Dockerfile +++ b/build/Dockerfile @@ -1,15 +1,16 @@ FROM golang:1.22-alpine as builder -ARG CGO_ENABLED=0 WORKDIR /app COPY go.mod go.sum ./ RUN go mod download + COPY . . -RUN go build cmd/proxy -o /out/app +RUN go build -o /out/app ./cmd/proxy -# multi stage build, to reduce size of the image FROM alpine:latest + COPY --from=builder /out/app /app + ENTRYPOINT ["/app"] \ No newline at end of file diff --git a/makefile b/makefile index bb14ce7..4f97745 100644 --- a/makefile +++ b/makefile @@ -1,5 +1,8 @@ +IMAGE_NAME =? /: + certs_windows: mkdir -p "./cmd/proxy/.cert" + # Create CA (certificate authority) openssl ecparam -out ./cmd/proxy/.cert/ca.key -name prime256v1 -genkey openssl req -new -sha256 -key ./cmd/proxy/.cert/ca.key -out ./cmd/proxy/.cert/ca.csr @@ -13,6 +16,7 @@ certs_windows: certs: mkdir -p "./cmd/proxy/.cert" + # Create CA (certificate authority) openssl ecparam -out ./cmd/proxy/.cert/ca.key -name prime256v1 -genkey openssl req -new -sha256 -key ./cmd/proxy/.cert/ca.key -out ./cmd/proxy/.cert/ca.csr @@ -25,4 +29,10 @@ certs: openssl x509 -in ./cmd/proxy/.cert/server.crt -text -noout mocks: - mockery \ No newline at end of file + mockery + +docker-build: + docker build -t ${IMAGE_NAME} -f .\build\Dockerfile . + +docker-push: + docker push -t ${IMAGE_NAME} -f .\build\Dockerfile . From 14cda9c81b66b10c4c8ae1ddf6681ea8caa5fe43 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:22:57 +0200 Subject: [PATCH 03/27] feat: Added Taskfile for the owner to compare it with Makefile --- Taskfile.yaml | 36 ++++++++++++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) create mode 100644 Taskfile.yaml diff --git a/Taskfile.yaml b/Taskfile.yaml new file mode 100644 index 0000000..a53b617 --- /dev/null +++ b/Taskfile.yaml @@ -0,0 +1,36 @@ +version: '3' + +tasks: + certs_windows: + cmds: + - mkdir -p "./cmd/proxy/.cert" + - openssl ecparam -out ./cmd/proxy/.cert/ca.key -name prime256v1 -genkey + - openssl req -new -sha256 -key ./cmd/proxy/.cert/ca.key -out ./cmd/proxy/.cert/ca.csr + - openssl x509 -req -sha256 -days 3650 -in ./cmd/proxy/.cert/ca.csr -signkey ./cmd/proxy/.cert/ca.key -out ./cmd/proxy/.cert/ca.crt + - openssl ecparam -out ./cmd/proxy/.cert/server.key -name prime256v1 -genkey + - openssl req -new -sha256 -key ./cmd/proxy/.cert/server.key -out ./cmd/proxy/.cert/server.csr + - openssl x509 -req -in ./cmd/proxy/.cert/server.csr -CA ./cmd/proxy/.cert/ca.crt -CAkey ./cmd/proxy/.cert/ca.key -CAcreateserial -out ./cmd/proxy/.cert/server.crt -days 3650 -sha256 + - openssl x509 -in ./cmd/proxy/.cert/server.crt -text -noout + + certs: + cmds: + - mkdir -p "./cmd/proxy/.cert" + - openssl ecparam -out ./cmd/proxy/.cert/ca.key -name prime256v1 -genkey + - openssl req -new -sha256 -key ./cmd/proxy/.cert/ca.key -out ./cmd/proxy/.cert/ca.csr + - openssl x509 -req -sha256 -days 3650 -in ./cmd/proxy/.cert/ca.csr -signkey ./cmd/proxy/.cert/ca.key -out ./cmd/proxy/.cert/ca.crt + - openssl ecparam -out ./cmd/proxy/.cert/server.key -name prime256v1 -genkey + - openssl req -new -sha256 -key ./cmd/proxy/.cert/server.key -out ./cmd/proxy/.cert/server.csr + - openssl x509 -req -in ./cmd/proxy/.cert/server.csr -CA ./cmd/proxy/.cert/ca.crt -CAkey ./cmd/proxy/.cert/ca.key -CAcreateserial -out ./cmd/proxy/.cert/server.crt -days 3650 -sha256 + - openssl x509 -in ./cmd/proxy/.cert/server.crt -text -noout + + mocks: + cmds: + - mockery + + docker-build: + cmds: + - docker build -t ${IMAGE_NAME} -f .\build\Dockerfile . + + docker-push: + cmds: + - docker push ${IMAGE_NAME} From 097300a60f9f9f4dbd63e2c8d2a9f3806de10db7 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:33:29 +0200 Subject: [PATCH 04/27] fix: Removed redundant clock package, changed all calls to time.Now() --- cmd/tcpproxy/main.go | 2 +- internal/ratelimit/rate.go | 4 +--- internal/ratelimit/rate_test.go | 6 ++---- internal/request/wrapper.go | 4 +--- 4 files changed, 5 insertions(+), 11 deletions(-) diff --git a/cmd/tcpproxy/main.go b/cmd/tcpproxy/main.go index 31e6095..4807f1e 100644 --- a/cmd/tcpproxy/main.go +++ b/cmd/tcpproxy/main.go @@ -48,7 +48,7 @@ func tcpDummy() error { for { conn, err := listener.Accept() if err != nil { - return fmt.Errorf("liistener accept: %w", err) + return fmt.Errorf("listener accept: %w", err) } log.Println(conn) diff --git a/internal/ratelimit/rate.go b/internal/ratelimit/rate.go index a1dedc3..d3ad749 100644 --- a/internal/ratelimit/rate.go +++ b/internal/ratelimit/rate.go @@ -3,8 +3,6 @@ package ratelimit import ( "net" "time" - - "waffle/internal/clock" ) type Rate struct { @@ -14,5 +12,5 @@ type Rate struct { } func (r *Rate) IsLimited() bool { - return clock.Now().Before(r.LimitedUntil) + return time.Now().Before(r.LimitedUntil) } diff --git a/internal/ratelimit/rate_test.go b/internal/ratelimit/rate_test.go index 427218c..41a056e 100644 --- a/internal/ratelimit/rate_test.go +++ b/internal/ratelimit/rate_test.go @@ -4,13 +4,11 @@ import ( "net" "testing" "time" - - "waffle/internal/clock" ) func TestRate_IsLimited(t *testing.T) { - limitedUntilBefore := clock.Now().Add(time.Duration(-1) * time.Hour) - limitedUntilAfter := clock.Now().Add(time.Hour) + limitedUntilBefore := time.Now().Add(time.Duration(-1) * time.Hour) + limitedUntilAfter := time.Now().Add(time.Hour) type fields struct { UUID string diff --git a/internal/request/wrapper.go b/internal/request/wrapper.go index f52c536..468a880 100644 --- a/internal/request/wrapper.go +++ b/internal/request/wrapper.go @@ -4,8 +4,6 @@ import ( "net" "net/http" "time" - - "waffle/internal/clock" ) // Wrapper is used to wrap request, in order to not pass it every time. @@ -21,6 +19,6 @@ func NewRequestWrapper(r *http.Request, ipAddress *net.IP) *Wrapper { return &Wrapper{ Request: r, IPAddress: ipAddress, - IssuedAt: clock.Now(), + IssuedAt: time.Now(), } } From 292d214ddc11e8ea50e02d58634fc32425065585 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:33:52 +0200 Subject: [PATCH 05/27] Completely removed clock package --- internal/clock/time.go | 9 --------- 1 file changed, 9 deletions(-) delete mode 100644 internal/clock/time.go diff --git a/internal/clock/time.go b/internal/clock/time.go deleted file mode 100644 index bd0d9ec..0000000 --- a/internal/clock/time.go +++ /dev/null @@ -1,9 +0,0 @@ -package clock - -import "time" - -var Now = nowFunc - -func nowFunc() time.Time { - return time.Now() -} From 4f24899b8c815bc89a77091d90f0a0ff8f770f12 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:41:16 +0200 Subject: [PATCH 06/27] refactor: Adding more descriptive error messages, and renaming vars --- cmd/collector/main.go | 10 +++++++--- internal/worker/collector.go | 4 ++-- 2 files changed, 9 insertions(+), 5 deletions(-) diff --git a/cmd/collector/main.go b/cmd/collector/main.go index 33e1acf..79026bf 100644 --- a/cmd/collector/main.go +++ b/cmd/collector/main.go @@ -20,11 +20,15 @@ func main() { ) flag.StringVar(&networkInterface, "i", defaultNetworkInterfaceDescription, "Identification of the interface") + // question: Why do we need context here? It is not used in collector.Run, except of ctx.Done, but since it is not + // context.WithTimeout (as example) it can not be closed in any way. + // Same in c.serializer.SerializePackets(ctx, packetsChan), it can not be closed there as well. + // Why not just to remove it? ctx := context.Background() log.Println("starting collector") - inMemoryPacketSerializer := packet.NewMemoryPacketSerializer(time.Minute * 5) + packetSerializer := packet.NewMemoryPacketSerializer(time.Minute * 5) // NEXT TODO: add BPF filter builder // https://www.ibm.com/docs/en/qsip/7.4?topic=queries-berkeley-packet-filters @@ -35,9 +39,9 @@ func main() { collector := worker.NewCollector( cfg, packet.NewWindowsNetworkInterfaceProvider(networkInterface), - inMemoryPacketSerializer) + packetSerializer) if err := collector.Run(ctx); err != nil { - panic(err.Error()) + log.Fatalln("Error during running collector: ", err.Error()) } } diff --git a/internal/worker/collector.go b/internal/worker/collector.go index ed5efd1..d796a44 100644 --- a/internal/worker/collector.go +++ b/internal/worker/collector.go @@ -91,7 +91,7 @@ func (c *Collector) Run(ctx context.Context) error { go func() { if err := c.serializer.SerializePackets(ctx, packetsChan); err != nil { - log.Println("error in serialize packets") + log.Printf("Error in serialize packets: %v\n", err) } }() @@ -99,7 +99,7 @@ func (c *Collector) Run(ctx context.Context) error { select { case packet, ok := <-packetSource.Packets(): if !ok { - log.Println("error reading packet") + log.Printf("Error while reading packet: %v\n", err) } packetsChan <- packet From 06ba9e1639f3231e6172581a6aa64334364a644d Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:55:02 +0200 Subject: [PATCH 07/27] Made visualizer and proxy server ports as flags, to allow user specify different ports --- cmd/proxy/main.go | 12 ++++++++++-- cmd/proxy/server/server.go | 15 ++++++++++----- 2 files changed, 20 insertions(+), 7 deletions(-) diff --git a/cmd/proxy/main.go b/cmd/proxy/main.go index a5c125e..cee6c4b 100644 --- a/cmd/proxy/main.go +++ b/cmd/proxy/main.go @@ -4,9 +4,10 @@ import ( "context" "embed" _ "embed" + "flag" "fmt" "os" - + "waffle/cmd/proxy/server" ) @@ -17,8 +18,15 @@ var yamlConfigBytes []byte var certificates embed.FS func main() { + var ( + visualizeServerPort string + proxyServerPort string + ) + flag.StringVar(&visualizeServerPort, "p", "8081", "Port for server to listen on") + flag.StringVar(&proxyServerPort, "p", "8081", "Port for server to listen on") + ctx := context.Background() - if err := server.Run(ctx, yamlConfigBytes, certificates); err != nil { + if err := server.Run(ctx, proxyServerPort, visualizeServerPort, yamlConfigBytes, certificates); err != nil { _, _ = fmt.Fprintf(os.Stderr, "%s\n", err) os.Exit(1) } diff --git a/cmd/proxy/server/server.go b/cmd/proxy/server/server.go index e1ae2ec..1fc76f2 100644 --- a/cmd/proxy/server/server.go +++ b/cmd/proxy/server/server.go @@ -3,6 +3,7 @@ package server import ( "context" "embed" + "fmt" "log" "os" "os/signal" @@ -41,7 +42,7 @@ import ( // If the proxy server fails to start, the function logs a fatal error. // // The function returns nil upon normal completion. -func Run(ctx context.Context, yamlConfigBytes []byte, certificates embed.FS) error { +func Run(ctx context.Context, proxyServerPort, visualizeServerPort string, yamlConfigBytes []byte, certificates embed.FS) error { _, cancel := signal.NotifyContext(ctx, os.Interrupt) defer cancel() @@ -67,18 +68,22 @@ func Run(ctx context.Context, yamlConfigBytes []byte, certificates embed.FS) err limiter := ratelimit.NewInMemoryLimiter(time.Minute * 5) - visualizeServer := visualize.NewServer(":8081") + visualizeServerPort = fmt.Sprintf(":%s", visualizeServerPort) + + s := visualize.NewServer(visualizeServerPort) guardHandler := waf.NewHandler( redirect.NewHandler(yamlDnsProvider), defender, limiter, - visualizeServer.GetVisualizer(), + s.GetVisualizer(), ) - proxyServer := proxy.NewServer(":8080", certificateProvider, guardHandler) + proxyServerPort = fmt.Sprintf(":%s", proxyServerPort) + + proxyServer := proxy.NewServer(proxyServerPort, certificateProvider, guardHandler) - log.Println("Starting Waffle Proxy on port :8080 🚀") + log.Printf("Starting Waffle Proxy on port %s 🚀\n", proxyServerPort) if err := proxyServer.Start(); err != nil { log.Fatal(err.Error()) From 317e5c78e1eb6b8aa12bf3d90ae974ef7db3f546 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:58:39 +0200 Subject: [PATCH 08/27] Handling errors during certificate loading. --- cmd/proxy/server/server.go | 49 ++++++++++++++++++++++++++++---------- 1 file changed, 37 insertions(+), 12 deletions(-) diff --git a/cmd/proxy/server/server.go b/cmd/proxy/server/server.go index 1fc76f2..1a8ab59 100644 --- a/cmd/proxy/server/server.go +++ b/cmd/proxy/server/server.go @@ -58,10 +58,26 @@ func Run(ctx context.Context, proxyServerPort, visualizeServerPort string, yamlC yamlDnsProvider := domain.NewYamlNameSystemProvider(yamlCfg) + caCerts, err := loadLocalCustomCACerts(certificates), + if err != nil { + return err + } + + certPemBlock, err := loadLocalCertPEMBlock(certificates), + if err != nil { + return err + } + + keyPemBlock, err := loadLocalKeyPEMBlock(certificates), + if err != nil { + return err + } + + certificateProvider := certificate.NewLocalCertificatesProvider( - loadLocalCustomCACerts(certificates), - loadLocalCertPEMBlock(certificates), - loadLocalKeyPEMBlock(certificates), + caCerts, + certPemBlock, + keyPemBlock, ) defender := guard.NewDefenseCoordinator([]guard.Defender{&guard.XSS{}}) @@ -95,21 +111,30 @@ func Run(ctx context.Context, proxyServerPort, visualizeServerPort string, yamlC // loadLocalCustomCACerts reads the local custom CA certificates from the embedded file system. // It reads the CA certificate file (ca.crt) located in the ".cert" directory and returns it as a slice of byte slices. // This CA certificate is used for establishing trust during TLS/SSL handshakes. -func loadLocalCustomCACerts(certificates embed.FS) [][]byte { - certBytes, _ := certificates.ReadFile(".cert/ca.crt") - return [][]byte{certBytes} +func loadLocalCustomCACerts(certificates embed.FS) ([][]byte, error) { + certBytes, err := certificates.ReadFile(".cert/ca.crt") + if err != nil { + return nil, err + } + return [][]byte{certBytes}, nil } // loadLocalCertPEMBlock reads the local server certificate (server.crt) from the embedded file system. // It returns the certificate as a byte slice, which is later used to serve the server's public certificate in TLS/SSL connections. -func loadLocalCertPEMBlock(certificates embed.FS) []byte { - certBytes, _ := certificates.ReadFile(".cert/server.crt") - return certBytes +func loadLocalCertPEMBlock(certificates embed.FS) ([]byte, error) { + certBytes, err := certificates.ReadFile(".cert/server.crt") + if err != nil { + return nil, err + } + return certBytes, nil } // loadLocalKeyPEMBlock reads the private key (server.key) from the embedded file system. // It returns the private key as a byte slice, which is paired with the server certificate during TLS/SSL handshakes. -func loadLocalKeyPEMBlock(certificates embed.FS) []byte { - certBytes, _ := certificates.ReadFile(".cert/server.key") - return certBytes +func loadLocalKeyPEMBlock(certificates embed.FS) ([]byte, error) { + certBytes, err := certificates.ReadFile(".cert/server.key") + if err != nil { + return nil, err + } + return certBytes, nil } From 9cb3ba96a245ac551751bbebb1c6ef331f616fd8 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 12:59:05 +0200 Subject: [PATCH 09/27] Fix: removed redundant commans at the end of line. --- cmd/proxy/server/server.go | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/cmd/proxy/server/server.go b/cmd/proxy/server/server.go index 1a8ab59..c56101e 100644 --- a/cmd/proxy/server/server.go +++ b/cmd/proxy/server/server.go @@ -58,22 +58,21 @@ func Run(ctx context.Context, proxyServerPort, visualizeServerPort string, yamlC yamlDnsProvider := domain.NewYamlNameSystemProvider(yamlCfg) - caCerts, err := loadLocalCustomCACerts(certificates), + caCerts, err := loadLocalCustomCACerts(certificates) if err != nil { return err } - certPemBlock, err := loadLocalCertPEMBlock(certificates), + certPemBlock, err := loadLocalCertPEMBlock(certificates) if err != nil { return err } - keyPemBlock, err := loadLocalKeyPEMBlock(certificates), + keyPemBlock, err := loadLocalKeyPEMBlock(certificates) if err != nil { return err } - certificateProvider := certificate.NewLocalCertificatesProvider( caCerts, certPemBlock, From 1ec018050e0ecf3c50a8ed69de915bc670d1fa48 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Mon, 21 Oct 2024 13:06:04 +0200 Subject: [PATCH 10/27] Handlded error in proxy/main.go --- cmd/proxy/main.go | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/cmd/proxy/main.go b/cmd/proxy/main.go index cee6c4b..481463a 100644 --- a/cmd/proxy/main.go +++ b/cmd/proxy/main.go @@ -5,9 +5,7 @@ import ( "embed" _ "embed" "flag" - "fmt" - "os" - + "log" "waffle/cmd/proxy/server" ) @@ -27,7 +25,6 @@ func main() { ctx := context.Background() if err := server.Run(ctx, proxyServerPort, visualizeServerPort, yamlConfigBytes, certificates); err != nil { - _, _ = fmt.Fprintf(os.Stderr, "%s\n", err) - os.Exit(1) + log.Fatalln(err) } } From 581314b9624437e2f9acaf2eea53f90a982274d8 Mon Sep 17 00:00:00 2001 From: werniq Date: Tue, 22 Oct 2024 09:28:33 +0200 Subject: [PATCH 11/27] Implemented graceful shutdown in collector and proxy --- cmd/collector/main.go | 9 +- cmd/proxy/main.go | 27 +++- cmd/proxy/server/server.go | 12 +- go.mod | 12 +- go.sum | 290 +---------------------------------- internal/worker/collector.go | 36 +++-- 6 files changed, 73 insertions(+), 313 deletions(-) diff --git a/cmd/collector/main.go b/cmd/collector/main.go index 79026bf..2326f92 100644 --- a/cmd/collector/main.go +++ b/cmd/collector/main.go @@ -1,7 +1,6 @@ package main import ( - "context" "flag" "log" "time" @@ -20,12 +19,6 @@ func main() { ) flag.StringVar(&networkInterface, "i", defaultNetworkInterfaceDescription, "Identification of the interface") - // question: Why do we need context here? It is not used in collector.Run, except of ctx.Done, but since it is not - // context.WithTimeout (as example) it can not be closed in any way. - // Same in c.serializer.SerializePackets(ctx, packetsChan), it can not be closed there as well. - // Why not just to remove it? - ctx := context.Background() - log.Println("starting collector") packetSerializer := packet.NewMemoryPacketSerializer(time.Minute * 5) @@ -41,7 +34,7 @@ func main() { packet.NewWindowsNetworkInterfaceProvider(networkInterface), packetSerializer) - if err := collector.Run(ctx); err != nil { + if err := collector.Run(); err != nil { log.Fatalln("Error during running collector: ", err.Error()) } } diff --git a/cmd/proxy/main.go b/cmd/proxy/main.go index 481463a..d0d3e34 100644 --- a/cmd/proxy/main.go +++ b/cmd/proxy/main.go @@ -6,6 +6,10 @@ import ( _ "embed" "flag" "log" + "os" + "os/signal" + "syscall" + "time" "waffle/cmd/proxy/server" ) @@ -23,8 +27,25 @@ func main() { flag.StringVar(&visualizeServerPort, "p", "8081", "Port for server to listen on") flag.StringVar(&proxyServerPort, "p", "8081", "Port for server to listen on") - ctx := context.Background() - if err := server.Run(ctx, proxyServerPort, visualizeServerPort, yamlConfigBytes, certificates); err != nil { - log.Fatalln(err) + quit := make(chan os.Signal) + + signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) + + go func() { + if err := server.Run(proxyServerPort, visualizeServerPort, yamlConfigBytes, certificates); err != nil { + log.Fatalln(err) + } + }() + + log.Println("Server started on :8080") + + <-quit + log.Println("Shutdown signal received, shutting down gracefully...") + + ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second) + defer cancel() + + if err := server.Shutdown(ctx); err != nil { + log.Fatalf("Server forced to shutdown: %s\n", err) } } diff --git a/cmd/proxy/server/server.go b/cmd/proxy/server/server.go index c56101e..5f4e8f1 100644 --- a/cmd/proxy/server/server.go +++ b/cmd/proxy/server/server.go @@ -5,8 +5,6 @@ import ( "embed" "fmt" "log" - "os" - "os/signal" "time" "waffle/internal/visualize" @@ -42,10 +40,7 @@ import ( // If the proxy server fails to start, the function logs a fatal error. // // The function returns nil upon normal completion. -func Run(ctx context.Context, proxyServerPort, visualizeServerPort string, yamlConfigBytes []byte, certificates embed.FS) error { - _, cancel := signal.NotifyContext(ctx, os.Interrupt) - defer cancel() - +func Run(proxyServerPort, visualizeServerPort string, yamlConfigBytes []byte, certificates embed.FS) error { _, err := config.LoadEnvironmentConfig() if err != nil { log.Fatal(err.Error()) @@ -107,6 +102,11 @@ func Run(ctx context.Context, proxyServerPort, visualizeServerPort string, yamlC return nil } +// Shutdown is a function to be called while 'gracefully shutdowning' the server. +func Shutdown(ctx context.Context) error { + return nil +} + // loadLocalCustomCACerts reads the local custom CA certificates from the embedded file system. // It reads the CA certificate file (ca.crt) located in the ".cert" directory and returns it as a slice of byte slices. // This CA certificate is used for establishing trust during TLS/SSL handshakes. diff --git a/go.mod b/go.mod index 4a6ec26..1850a15 100644 --- a/go.mod +++ b/go.mod @@ -4,29 +4,31 @@ go 1.22.0 require ( github.com/Netflix/go-env v0.0.0-20220526054621-78278af1949d + github.com/cdipaolo/goml v0.0.0-20220715001353-00e0c845ae1c github.com/corazawaf/libinjection-go v0.1.3 github.com/emirpasic/gods v1.18.1 github.com/goccy/go-yaml v1.11.3 + github.com/google/gopacket v1.1.19 github.com/google/uuid v1.6.0 + github.com/malaschitz/randomForest v0.0.0-20240228214944-c64ffe1648e0 github.com/patrickmn/go-cache v2.1.0+incompatible github.com/stretchr/testify v1.9.0 + golang.org/x/sys v0.17.0 nhooyr.io/websocket v1.8.10 ) require ( - github.com/cdipaolo/goml v0.0.0-20220715001353-00e0c845ae1c // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/fatih/color v1.16.0 // indirect - github.com/google/gopacket v1.1.19 // indirect - github.com/malaschitz/randomForest v0.0.0-20240228214944-c64ffe1648e0 // indirect github.com/mattn/go-colorable v0.1.13 // indirect github.com/mattn/go-isatty v0.0.20 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect - github.com/sjwhitworth/golearn v0.0.0-20221228163002-74ae077eafb2 // indirect github.com/stretchr/objx v0.5.2 // indirect golang.org/x/crypto v0.14.0 // indirect - golang.org/x/sys v0.17.0 // indirect + golang.org/x/net v0.10.0 // indirect + golang.org/x/text v0.13.0 // indirect golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028 // indirect gonum.org/v1/gonum v0.9.3 // indirect + gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index 04b52f2..48ca51e 100644 --- a/go.sum +++ b/go.sum @@ -1,55 +1,22 @@ -bazil.org/fuse v0.0.0-20160811212531-371fbbdaa898/go.mod h1:Xbm+BRKSBEpa4q4hTSxohYNQpsxXPbPry4JJWOB3LB8= cloud.google.com/go v0.26.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.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= -cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= -cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= -cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= -cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0= -cloud.google.com/go v0.49.0/go.mod h1:hGvAdzcWNbyuxS3nWhD7H2cIJxjRRTRLQVB0bdputVY= -cloud.google.com/go v0.50.0/go.mod h1:r9sluTvynVuxRIOHXQEHMFffphuXHOMZMycpNR5e6To= -cloud.google.com/go v0.52.0/go.mod h1:pXajvRH/6o3+F9jDHZWQ5PbGhn+o8w9qiu/CffaVdO4= -cloud.google.com/go v0.53.0/go.mod h1:fp/UouUEsRkN6ryDKNW/Upv/JBKnv6WDthjR6+vze6M= -cloud.google.com/go v0.54.0/go.mod h1:1rq2OEkV3YMf6n/9ZvGWI3GWw0VoqH/1x2nd8Is/bPc= -cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= -cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= -cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE= -cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc= -cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= -cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= -cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= -cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= -cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA= -cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw= -cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0ZeosJ0Rtdos= -cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohlUTyfDhBk= dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= dmitri.shuralyov.com/gpu/mtl v0.0.0-20201218220906-28db891af037/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= gioui.org v0.0.0-20210308172011-57750fc8a0a6/go.mod h1:RSH6KIUZ0p2xy5zHDxgAM4zumjgTw83q2ge/PI+yyw8= -github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= -github.com/DATA-DOG/go-sqlmock v1.3.3/go.mod h1:f/Ixk793poVmq4qj/V1dPUg2JEAKC73Q5eFN3EC/SaM= -github.com/DataDog/datadog-go v0.0.0-20180822151419-281ae9f2d895/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= -github.com/DzananGanic/numericalgo v0.0.0-20170804125527-2b389385baf0/go.mod h1:uIo7VpFvBkDQoCyKqUL/mTNjpOlv1KdWaJyCsBSpCe4= github.com/Knetic/govaluate v3.0.1-0.20171022003610-9aa49832a739+incompatible/go.mod h1:r7JcOSlj0wfOMncg0iLm8Leh48TZaKVeNIfJntJ2wa0= -github.com/Microsoft/go-winio v0.4.14/go.mod h1:qXqCSQ3Xa7+6tgxaGTIe4Kpcdsi+P8jBhyzoq1bpyYA= -github.com/NYTimes/gziphandler v1.1.1/go.mod h1:n/CVRwUEOgIxrgPvAQhUUr9oeUtvrhMomdKFjzJNB0c= github.com/Netflix/go-env v0.0.0-20220526054621-78278af1949d h1:wvStE9wLpws31NiWUx+38wny1msZ/tm+eL5xmm4Y7So= github.com/Netflix/go-env v0.0.0-20220526054621-78278af1949d/go.mod h1:9XMFaCeRyW7fC9XJOWQ+NdAv8VLG7ys7l3x4ozEGLUQ= -github.com/Nvveen/Gotty v0.0.0-20120604004816-cd527374f1e5/go.mod h1:lmUJ/7eu/Q8D7ML55dXQrVaamCz2vxCfdQBasLZfHKk= github.com/Shopify/sarama v1.19.0/go.mod h1:FVkBWblsNy7DGZRfXLU0O9RCGt5g3g3yEuWXgklEdEo= github.com/Shopify/toxiproxy v2.1.4+incompatible/go.mod h1:OXgGpZ6Cli1/URJOF1DMxUHB2q5Ap20/P/eIdh4G0pI= github.com/VividCortex/gohistogram v1.0.0/go.mod h1:Pf5mBqqDxYaXu3hDrrU+w6nw50o/4+TcAqDqk/vUH7g= github.com/afex/hystrix-go v0.0.0-20180502004556-fa1af6a1f4f5/go.mod h1:SkGFH1ia65gfNATL8TAiHDNxPzPdmEL5uirI2Uyuz6c= -github.com/airbrake/gobrake v3.6.1+incompatible/go.mod h1:wM4gu3Cn0W0K7GUuVWnlXZU11AGBXMILnrdOU8Kn00o= github.com/ajstarks/svgo v0.0.0-20180226025133-644b8db467af/go.mod h1:K08gAheRH3/J6wwsYMMT4xOr94bZjxIelGM0+d/wbFw= 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.0.0-20181112125854-24918abba929/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= github.com/apache/thrift v0.12.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= github.com/apache/thrift v0.13.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= @@ -58,40 +25,26 @@ github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj github.com/aryann/difflib v0.0.0-20170710044230-e206f873d14a/go.mod h1:DAHtR1m6lCRdSC2Tm3DSWRPvIPr6xNKyeHdqDQSQT+A= github.com/aws/aws-lambda-go v1.13.3/go.mod h1:4UKl9IzQMoD+QF79YdCuzCwp8VbmG4VAQwij/eHl5CU= github.com/aws/aws-sdk-go v1.27.0/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= -github.com/aws/aws-sdk-go v1.30.19/go.mod h1:5zCpMtNQVjRREroY7sYe8lOMRSxkhG6MZveU8YkpAk0= github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= 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/blend/go-sdk v1.1.1/go.mod h1:IP1XHXFveOXHRnojRJO7XvqWGqyzevtXND9AdSztAe8= github.com/boombuler/barcode v1.0.0/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= -github.com/bradfitz/go-smtpd v0.0.0-20170404230938-deb6d6237625/go.mod h1:HYsPBTaaSFSlLx/70C2HPIMNZpVV8+vt/A+FMnYP11g= -github.com/brianvoe/gofakeit/v4 v4.3.0/go.mod h1:GC/GhKWdGJ2eskBf4zGdjo3eHj8rX4E9hFLFg0bqK4s= github.com/casbin/casbin/v2 v2.1.2/go.mod h1:YcPU1XXisHhLzuxH9coDNf2FbKpjGlbCg3n9yuLkIJQ= github.com/cdipaolo/goml v0.0.0-20220715001353-00e0c845ae1c h1:uqJXOhayPfl/QruVBP6VF0KUWNDzO/F14X8CPEkkFD8= github.com/cdipaolo/goml v0.0.0-20220715001353-00e0c845ae1c/go.mod h1:Ue8jgVLdBDCtsh1laikvraXqXzKCyKiruCcCcaeNDFE= github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= -github.com/cenkalti/backoff/v4 v4.0.2/go.mod h1:eEew/i+1Q6OrCDZh3WiXYv3+nJwBASZ8Bog/87DQnVg= 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/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/clbanning/x2j v0.0.0-20191024224557-825249438eec/go.mod h1:jMjuTZXRI4dUb/I5gc9Hdhagfvm9+RyrPryS/auMzxE= 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/cnkei/gospline v0.0.0-20191204072713-842a72f86331/go.mod h1:DXXGDL64/wxXgBSgmGMEL0vYC0tdvpgNhkJrvavhqDM= github.com/cockroachdb/datadriven v0.0.0-20190809214429-80d97fb3cbaa/go.mod h1:zn76sxSg3SzpJ0PPJaLDCu+Bu0Lg3sKTORVIj19EIF8= github.com/codahale/hdrhistogram v0.0.0-20161010025455-3a0bb77429bd/go.mod h1:sE/e/2PUdi/liOCUjSTXgM1o87ZssimdTWN964YiIeI= -github.com/colinmarc/hdfs/v2 v2.1.1/go.mod h1:M3x+k8UKKmxtFu++uAZ0OtDU8jR3jnaZIAc6yK4Ue0c= -github.com/containerd/continuity v0.0.0-20191127005431-f65d91d395eb/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y= -github.com/containerd/continuity v0.0.0-20200413184840-d3ef23f19fbb/go.mod h1:Dq467ZllaHgAtVp4p1xUQWBrFXR9s/wyoTpG8zOJGkY= github.com/corazawaf/libinjection-go v0.1.3 h1:PUplAYho1BBl0tIVbhDsNRuVGIeUYSiCEc9oQpb2rJU= github.com/corazawaf/libinjection-go v0.1.3/go.mod h1:OP4TM7xdJ2skyXqNX1AN1wN5nNZEmJNuWbNPOItn7aw= github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= github.com/coreos/go-systemd v0.0.0-20180511133405-39ca1b05acc7/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= -github.com/coreos/go-systemd v0.0.0-20181012123002-c6f51f82210d/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= @@ -100,8 +53,6 @@ github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSs 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/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= -github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec= -github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= github.com/dustin/go-humanize v0.0.0-20171111073723-bb3d318650d4/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= github.com/eapache/go-resiliency v1.1.0/go.mod h1:kFI+JgMyC7bLPUVY133qvEBtVayf5mFgVsvEsIPBvNs= github.com/eapache/go-xerial-snappy v0.0.0-20180814174437-776d5712da21/go.mod h1:+020luEh2TKB4/GOp8oxxtq0Daoen/Cii55CzbTV6DU= @@ -110,28 +61,22 @@ github.com/edsrzf/mmap-go v1.0.0/go.mod h1:YO35OhQPt3KJa3ryjFM5Bs14WD66h8eGKpfaB github.com/emirpasic/gods v1.18.1 h1:FXtiHYKDGKCW2KzwZKx0iC0PQmdlorYgdFG9jPXJ1Bc= github.com/emirpasic/gods v1.18.1/go.mod h1:8tpGGwCnJ5H4r6BWwaV6OrWmMoPhUl5jm/FMNAnJvWQ= github.com/envoyproxy/go-control-plane v0.6.9/go.mod h1:SBwIajubJHhxtWwsL9s8ss4safvEdbitLhGGK48rN6g= -github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/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/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= github.com/fatih/color v1.16.0 h1:zmkK9Ngbjj+K0yRhTVONQh1p/HknKYSlNT+vZCzyokM= github.com/fatih/color v1.16.0/go.mod h1:fL2Sau1YI5c0pdGEVCbKQbLXB6edEj1ZgiY4NijnWvE= -github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= github.com/fogleman/gg v1.2.1-0.20190220221249-0403632d5b90/go.mod h1:R/bRT+9gY/C5z7JzPU0zXsXHKM4/ayA+zqcVNZzPa1k= github.com/fogleman/gg v1.3.0/go.mod h1:R/bRT+9gY/C5z7JzPU0zXsXHKM4/ayA+zqcVNZzPa1k= github.com/franela/goblin v0.0.0-20200105215937-c9ffbefa60db/go.mod h1:7dvUGVsVBjqR7JHJk0brhHOZYGmfBYOrK0ZhYMEtBr4= github.com/franela/goreq v0.0.0-20171204163338-bcd34c9993f8/go.mod h1:ZhphrRTfi2rbfLwlschooIH4+wKKDR4Pdxhh+TRoA20= -github.com/frankban/quicktest v1.5.0/go.mod h1:jaStnuzAqU1AJdCO0l53JDCJrVDKcS03DbaAcR7Ks/o= github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= 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-fonts/dejavu v0.1.0/go.mod h1:4Wt4I4OU2Nq9asgDCteaAaWZOV24E+0/Pwo0gppep4g= github.com/go-fonts/latin-modern v0.2.0/go.mod h1:rQVLdDMK+mK1xscDwsqM5J8U2jrRa3T0ecnM9pNujks= github.com/go-fonts/liberation v0.1.1/go.mod h1:K6qoJYypsmfVjWg8KOVDQhLc8UDgIK2HYqyqAO9z7GY= github.com/go-fonts/stix v0.1.0/go.mod h1:w/c1f0ldAUlJmLBvlbkvVXLAD+tAMqobIIQpmnUIzUY= github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= -github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/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= @@ -148,7 +93,6 @@ github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+ github.com/go-playground/validator/v10 v10.4.1 h1:pH2c5ADXtd66mxoE0Zm9SUhxE20r7aM3F26W0hOn+GE= github.com/go-playground/validator/v10 v10.4.1/go.mod h1:nlOn6nFhuKACm19sB/8EGNn9GlaMV7XkbRSipzJ0Ii4= github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= -github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= github.com/goccy/go-yaml v1.11.3 h1:B3W9IdWbvrUu2OYQGwvU1nZtvMQJPBKgBUuweJjLj6I= github.com/goccy/go-yaml v1.11.3/go.mod h1:wKnAMd44+9JAAnGQpWVEgBzGt3YuTaQ4uXoHvE4m7WU= @@ -160,67 +104,34 @@ github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGw github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= -github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= -github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= 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.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= -github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= -github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= -github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= -github.com/golang/protobuf v1.1.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= 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.3.4/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/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= -github.com/gonum/blas v0.0.0-20181208220705-f22b278b28ac/go.mod h1:P32wAyui1PQ58Oce/KYkOqQv8cVw1zAapXOl+dRFGbc= -github.com/gonum/lapack v0.0.0-20181123203213-e4cdc5a0bff9/go.mod h1:XA3DeT6rxh2EAE789SSiSJNqxPaC0aE9J8NTOI0Jo/A= -github.com/gonum/matrix v0.0.0-20181209220409-c518dec07be9/go.mod h1:0EXg4mc1CNP0HCqCz+K4ts155PXIlUywf0wqN+GfPZw= github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/btree v1.0.0/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.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= -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/gopacket v1.1.19 h1:ves8RnFZPGiFnTS0uPQStjwru6uO6h+nlr9j6fL7kF8= github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo= -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-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= -github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= github.com/google/uuid v1.0.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= -github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= github.com/gorilla/context v1.1.1/go.mod h1:kBGZzfjB9CEq2AlWe17Uuf7NDRt0dE0s8S51q0aT7Yg= github.com/gorilla/mux v1.6.2/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs= github.com/gorilla/mux v1.7.3/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs= github.com/gorilla/websocket v0.0.0-20170926233335-4201258b820c/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= -github.com/gotestyourself/gotestyourself v2.2.0+incompatible/go.mod h1:zZKM6oeNM8k+FRljX1mnzVYeS8wiGgQyvST1/GafPbY= -github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA= github.com/grpc-ecosystem/go-grpc-middleware v1.0.1-0.20190118093823-f849b5445de4/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk= github.com/grpc-ecosystem/grpc-gateway v1.9.5/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= -github.com/guptarohit/asciigraph v0.5.1/go.mod h1:9fYEfE5IGJGxlP1B+w8wHFy7sNZMhPtn59f0RLtpRFM= github.com/hashicorp/consul/api v1.3.0/go.mod h1:MmDNSzIMUjNpY/mQ398R4bk2FnqQLoPndWW5VkKPlCE= github.com/hashicorp/consul/sdk v0.3.0/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= @@ -231,7 +142,6 @@ github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHh github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU= github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= -github.com/hashicorp/go-uuid v0.0.0-20180228145832-27454136f036/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= github.com/hashicorp/go-version v1.2.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= @@ -244,38 +154,28 @@ github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2p github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= github.com/hudl/fargo v1.3.0/go.mod h1:y3CKSmjA+wD2gak7sUSXTAoopbhU08POFhmITJgmKTg= -github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= -github.com/icza/gox v0.0.0-20200320174535-a6ff52ab3d90/go.mod h1:VbcN86fRkkUMPX2ufM85Um8zFndLZswoIW1eYtpAcVk= github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= -github.com/jcmturner/gofork v0.0.0-20180107083740-2aebee971930/go.mod h1:MK8+TM0La+2rjBD4jE12Kj1pCCxK7d2LK/UM3ncEo0o= -github.com/jellevandenhooff/dkim v0.0.0-20150330215556-f50fe3d243e1/go.mod h1:E0B/fFc00Y+Rasa88328GlI/XbtyysCtTHZS8h7IrBU= github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= -github.com/jmespath/go-jmespath v0.3.0/go.mod h1:9QtRXoHjLGCJ5IBSaohpXITPlowMeeYCZ7fLUTSywik= -github.com/jmoiron/sqlx v1.2.0/go.mod h1:1FEQNm3xlJgrMD+FBdI9+xvCksHtbpVBBw5dYhBSsks= github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.8/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= -github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= github.com/jung-kurt/gofpdf v1.0.0/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= github.com/jung-kurt/gofpdf v1.0.3-0.20190309125859-24315acbbda5/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= -github.com/klauspost/compress v1.9.7/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A= 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 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= 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/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y= github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII= -github.com/lib/pq v1.0.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= github.com/lightstep/lightstep-tracer-common/golang/gogo v0.0.0-20190605223551-bc2310a04743/go.mod h1:qklhhLq1aX+mtWk9cPHPzaBjWImj5ULL6C7HFJtXQMM= github.com/lightstep/lightstep-tracer-go v0.18.1/go.mod h1:jlF1pusYV4pidLvZ+XD0UBX0ZE6WURAspgAczcDHrL4= github.com/lyft/protoc-gen-validate v0.0.13/go.mod h1:XbGvPuh87YZc5TdIa2/I4pLk0QoUACkjt2znoq26NVQ= @@ -291,8 +191,6 @@ github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/ github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= github.com/mattn/go-runewidth v0.0.2/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= -github.com/mattn/go-runewidth v0.0.7/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= -github.com/mattn/go-sqlite3 v1.9.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc= github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= @@ -317,17 +215,10 @@ github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OS github.com/oklog/oklog v0.3.2/go.mod h1:FCV+B7mhrz4o+ueLpx+KqkyXRGMWOYEvfiXtdGtbWGs= github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA= github.com/olekukonko/tablewriter v0.0.0-20170122224234-a0225b3f23b5/go.mod h1:vsDQFd/mU46D+Z4whnwzcISnGGzXWMclvtLoiIKAKIo= -github.com/olekukonko/tablewriter v0.0.4/go.mod h1:zq6QwlOf5SlnkVbMSr5EoBv3636FWnp+qbPhuoO21uA= -github.com/ompluscator/dynamic-struct v1.2.0/go.mod h1:ADQ1+6Ox1D+ntuNwTHyl1NvpAqY2lBXPSPbcO4CJdeA= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= -github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= -github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk= -github.com/opencontainers/go-digest v1.0.0-rc1/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s= -github.com/opencontainers/image-spec v1.0.1/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= -github.com/opencontainers/runc v0.1.1/go.mod h1:qT5XzbpPznkRYVz/mWwUaVBUv2rmF59PVA73FjuZG0U= github.com/opentracing-contrib/go-observer v0.0.0-20170622124052-a52f23424492/go.mod h1:Ngi6UdF0k5OKD5t5wlmGhe/EDKPoUM3BXZSSfIuJbis= github.com/opentracing/basictracer-go v1.0.0/go.mod h1:QfBfYuafItcjQuMwinw9GhYKwFXS9KnPs5lxoYwgW74= github.com/opentracing/opentracing-go v1.0.2/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= @@ -336,22 +227,18 @@ github.com/openzipkin-contrib/zipkin-go-opentracing v0.4.5/go.mod h1:/wsWhb9smxS github.com/openzipkin/zipkin-go v0.1.6/go.mod h1:QgAqvLzwWbR/WpD4A3cGpPtJrZXNIiJc5AZX7/PBEpw= github.com/openzipkin/zipkin-go v0.2.1/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4= github.com/openzipkin/zipkin-go v0.2.2/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4= -github.com/ory/dockertest v3.3.5+incompatible/go.mod h1:1vX4m9wsvi00u5bseYwXaSnhNrne+V0E6LAcBILJdPs= github.com/pact-foundation/pact-go v1.0.4/go.mod h1:uExwJY4kCzNPcHRj+hCR/HBbOOIwwtUjcrb0b5/5kLM= github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= github.com/patrickmn/go-cache v2.1.0+incompatible h1:HRMgzkcYKYpi3C8ajMPV8OFXaaRUnok+kx1WdO15EQc= github.com/patrickmn/go-cache v2.1.0+incompatible/go.mod h1:3Qf8kWWT7OJRJbdiICTKqZju1ZixQ/KpMGzzAfe6+WQ= -github.com/pborman/getopt v0.0.0-20180729010549-6fdd0a2c7117/go.mod h1:85jBQOZwpVEaDAr341tbn15RS4fCAsIst0qp7i8ex1o= github.com/pborman/uuid v1.2.0/go.mod h1:X/NO0urCmaxf9VXbdlT7C2Yzkj2IKimNn4k+gtPdI/k= github.com/performancecopilot/speed v3.0.0+incompatible/go.mod h1:/CLtqpZ5gBg1M9iaPbIdPPGyKcA8hKdoy6hAWba7Yac= github.com/petar/GoMNIST v0.0.0-20150320212226-2fbe10d0fa63/go.mod h1:d7fwuOuDrb75/3iplL4oWbe4MBZgWil/pSVR3ItECVU= -github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU= github.com/phpdave11/gofpdf v1.4.2/go.mod h1:zpO6xFn9yxo3YLyMvW8HcKWVdbNqgIfOOp2dXMnm1mY= github.com/phpdave11/gofpdi v1.0.12/go.mod h1:vBmVV0Do6hSBHC8uKUQ71JGW+ZGQq74llk/7bXwjDoI= github.com/pierrec/lz4 v1.0.2-0.20190131084431-473cd7ce01a1/go.mod h1:3/3N9NVKO0jef7pBehbT1qWhCMrIgbYNnFAZCqQ5LRc= 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-0.20171018195549-f15c970de5b7/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/profile v1.2.1/go.mod h1:hJw3o1OdXxsrSjjVksARp5W95eeEaEfptyVZyv6JUPA= @@ -375,11 +262,7 @@ github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1/go.mod h1:c3At6R 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-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= -github.com/rocketlaunchr/dataframe-go v0.0.0-20201007021539-67b046771f0b/go.mod h1:FsS1JF7xpC3WIxMu8DtEyxCNXl1SbHLTlUNE7QcETpA= -github.com/rocketlaunchr/dbq/v2 v2.5.0/go.mod h1:MckY8J697t+AGc0ENl968yDVnD5cP/FFOBSPPyJXY5A= -github.com/rocketlaunchr/mysql-go v1.1.3/go.mod h1:SD/1bpRrmcdnBYRJq8eCerqqS1nTR9Y9WdW+LPzDLAQ= github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= -github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/rs/xid v1.2.1/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ= github.com/rs/zerolog v1.21.0/go.mod h1:ZPhntP/xmq1nnND05hhpAh2QMhSsA4UN3MGZ6O2J3hM= @@ -387,26 +270,16 @@ github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQD github.com/ruudk/golang-pdf417 v0.0.0-20181029194003-1af4ab5afa58/go.mod h1:6lfFZQK844Gfx8o5WFuvpxWRwnSoipWe/p622j1v06w= github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= github.com/samuel/go-zookeeper v0.0.0-20190923202752-2cc03de413da/go.mod h1:gi+0XIa01GRL2eRQVjQkKGqKF3SF9vZR/HnPullcV2E= -github.com/sandertv/go-formula/v2 v2.0.0-alpha.7/go.mod h1:Ag4V2fiOHWXct3SraXNN3dFzFtyu9vqBfrjfYWMGLhE= github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= -github.com/shabbyrobe/xmlwriter v0.0.0-20200208144257-9fca06d00ffa/go.mod h1:Yjr3bdWaVWyME1kha7X0jsz3k2DgXNa1Pj3XGyUAbx8= github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= -github.com/sirupsen/logrus v1.0.4-0.20170822132746-89742aefa4b2/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc= github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= -github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q= 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/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= -github.com/sjwhitworth/golearn v0.0.0-20221228163002-74ae077eafb2 h1:wv0gCxjJAuQJDUlOLsjM/1QPq0VF3tR7n3cMkEf3q+I= -github.com/sjwhitworth/golearn v0.0.0-20221228163002-74ae077eafb2/go.mod h1:rrvYclvrqwEsURE+k7VH2nhOT6BV+IutaIgBBQ9Wdeg= github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM= github.com/sony/gobreaker v0.4.1/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY= -github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk= -github.com/spf13/cobra v0.0.2-0.20171109065643-2da4a54c5cee/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= -github.com/spf13/pflag v1.0.1-0.20171106142849-4c012f6dcd95/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= github.com/spf13/pflag v1.0.1/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= github.com/streadway/amqp v0.0.0-20190404075320-75d898a42a94/go.mod h1:AZpEONHx3DKn8O/DFsRAY58/XVQiIPMTMB1SddzLXVw= github.com/streadway/amqp v0.0.0-20190827072141-edfb9018d271/go.mod h1:AZpEONHx3DKn8O/DFsRAY58/XVQiIPMTMB1SddzLXVw= @@ -418,34 +291,20 @@ github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/ 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.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= -github.com/tarm/serial v0.0.0-20180830185346-98f6abe2eb07/go.mod h1:kDXzergiv9cbyO7IOYJZWg1U88JhDg3PB6klq9Hg2pA= -github.com/tealeg/xlsx/v3 v3.0.0/go.mod h1:fSua0Owrk9yAMAFGZI7piq5UL2BcubuQuLNOEhr3X80= github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= -github.com/wcharczuk/go-chart v2.0.1+incompatible/go.mod h1:PF5tmL4EIx/7Wf+hEkpCqYi5He4u90sw+0+6FhrryuE= github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= -github.com/xitongsys/parquet-go v1.5.1/go.mod h1:xUxwM8ELydxh4edHGegYq1pA8NnMKDx0K/GyB0o2bww= -github.com/xitongsys/parquet-go v1.5.2/go.mod h1:90swTgY6VkNM4MkMDsNxq8h30m6Yj1Arv9UMEl5V5DM= -github.com/xitongsys/parquet-go-source v0.0.0-20190524061010-2b72cbee77d5/go.mod h1:xxCx7Wpym/3QCo6JhujJX51dzSXrwmb0oH6FQb39SEA= -github.com/xitongsys/parquet-go-source v0.0.0-20200326031722-42b453e70c3b/go.mod h1:xxCx7Wpym/3QCo6JhujJX51dzSXrwmb0oH6FQb39SEA= -github.com/xitongsys/parquet-go-source v0.0.0-20200509081216-8db33acb0acf/go.mod h1:EVm7J5W7X/BJsvlGnCaj81kYxgbNzssi/+LF16FoV2s= -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.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= -github.com/zserge/lorca v0.1.9/go.mod h1:bVmnIbIRlOcoV285KIRSe4bUABKi7R7384Ycuum6e4A= go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg= go.opencensus.io v0.20.1/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= go.opencensus.io v0.20.2/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= -go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= -go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= -go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opentelemetry.io/otel v0.20.0/go.mod h1:Y3ugLH2oa81t5QO+Lty+zXf8zC9L26ax4Nzoxm/dooo= go.opentelemetry.io/otel/metric v0.20.0/go.mod h1:598I5tYlH1vzBjn+BTuhzTCSb/9debfNp6R3s7Pr1eU= go.opentelemetry.io/otel/oteltest v0.20.0/go.mod h1:L7bgKf9ZB7qCwT9Up7i9/pn0PWIa9FqQ2IQ8LoxiGnw= @@ -463,18 +322,12 @@ go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9E go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= go.uber.org/zap v1.13.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= go.uber.org/zap v1.16.0/go.mod h1:MA8QOfq0BHJwdXa996Y4dYkAqRKB8/1K1QMMZVaNZjQ= -go4.org v0.0.0-20180809161055-417644f6feb5/go.mod h1:MkTOUMDaeVYJUOUsaDXIhWPZYa1yOyC1qaOBpL57BhE= -golang.org/x/build v0.0.0-20200402160453-61705b562fc9/go.mod h1:BEpJH1kxLue/53k7XKPHBk7Qb3nvSlpB/rQWTi7bMdA= -golang.org/x/crypto v0.0.0-20171113213409-9f005a07e0d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= -golang.org/x/crypto v0.0.0-20180723164146-c126467f60eb/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= 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-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4= @@ -483,17 +336,9 @@ golang.org/x/exp v0.0.0-20180807140117-3d87b88a115f/go.mod h1:CJ0aWSM057203Lf6IL golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190125153040-c74c464bbbf2/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= -golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= golang.org/x/exp v0.0.0-20190731235908-ec7cb31e5a56/go.mod h1:JhuoJpWY28nO4Vef9tZUw9qufEGTyX1+7lmHxV5q5G4= -golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= golang.org/x/exp v0.0.0-20191002040644-a1355ae1e2c3/go.mod h1:NOZ3BPKG0ec/BKJQgnvsSFpcKLM5xXVWnvZS97DWHgE= -golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY= -golang.org/x/exp v0.0.0-20191129062945-2f5052295587/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= -golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= -golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= -golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM= -golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU= -golang.org/x/exp v0.0.0-20200331195152-e8c3332aa8e5/go.mod h1:4M0jN8W1tt0AVLNr8HDosyJCDCDuyL9N9+3m7wDWgKw= +golang.org/x/exp v0.0.0-20210916165020-5cb4fee858ee h1:qlrAyYdKz4o7rWVUjiKqQJMa4PEpd55fqBU8jpsl4Iw= golang.org/x/exp v0.0.0-20210916165020-5cb4fee858ee/go.mod h1:a3o/VtDNHN+dCVLEpzjjUHOzR+Ln3DHX056ZPzoZGGA= golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs= golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= @@ -508,11 +353,7 @@ golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTk golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs= -golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= @@ -520,9 +361,7 @@ golang.org/x/mobile v0.0.0-20201217150744-e6ae53a27f4f/go.mod h1:skQtrUTUwhdJvXM golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY= golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= -golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= golang.org/x/mod v0.1.1-0.20191209134235-331c550502dd/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -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-0.20210830214625-1b1db11ec8f4/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro= @@ -538,37 +377,22 @@ golang.org/x/net v0.0.0-20190125091013-d26f9f9a57f3/go.mod h1:mL1N/T3taQHkDXs73r 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-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= 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-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190813141303-74dc4d7220e7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/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-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200222125558-5a598a2470a0/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-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20200501053045-e0ff5e5a1de5/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.10.0 h1:X2//UzNDwYmtCLn7To6G58Wr6f5ahEAQgKNzv9Y951M= +golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/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/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/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-20200317015054-43a5402ce75a/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/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= @@ -584,24 +408,10 @@ golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7w 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-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191220142924-d4481acd189f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/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-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -613,22 +423,19 @@ golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.17.0 h1:25cE3gD+tdBA7lp7QfhuV+rJiE9YXTcS3VG1SqssI/Y= golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= 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.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.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= +golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= 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/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= 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-20181205014116-22934f0fdb62/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-20190206041539-40960b6deb8e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= @@ -636,37 +443,15 @@ golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3 golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= -golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= -golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= -golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= -golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= -golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20190927191325-030b2cf1153e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191029041327-9cc4af7d6b2c/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191029190741-b9c20aec41a5/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191115202509-3a792d9c32b2/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-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191130070609-6e064ea0cf2d/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191216173652-a0e659d51361/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200103221440-774c71fcf114/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200117012304-6edc0a871e69/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200117161641-43d50277825c/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200122220014-bf1340f18c4a/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200204074204-1cc6d1ef6c74/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200212150539-ea181f53ac56/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200224181240-023911ca70b2/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= -golang.org/x/tools v0.0.0-20200402223321-bcf690261a44/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= golang.org/x/tools v0.1.6/go.mod h1:LGqMHiF4EqQNHR1JncWGqT5BVaXmza+X+BDGol+dOxo= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -676,77 +461,33 @@ golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8T golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028 h1:+cNy6SZtPcJQH3LJVLOSmiC7MMxXNOb3PU/VUEz+EhU= golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028/go.mod h1:NDW/Ps6MPRej6fsCIbMTohpP40sJ/P/vI1MoTEGwX90= gonum.org/v1/gonum v0.0.0-20180816165407-929014505bf4/go.mod h1:Y+Yx5eoAFn32cQvJDxZx5Dpnq+c3wtXuadVZAcxbbBo= -gonum.org/v1/gonum v0.7.0/go.mod h1:L02bwd0sqlsvRv41G7wGWFCsVNZFv/k1xzGIxeANHGM= -gonum.org/v1/gonum v0.8.1/go.mod h1:oe/vMfY3deqTw+1EZJhuvEW2iwGF1bW9wwu7XCu0+v0= gonum.org/v1/gonum v0.8.2/go.mod h1:oe/vMfY3deqTw+1EZJhuvEW2iwGF1bW9wwu7XCu0+v0= gonum.org/v1/gonum v0.9.3 h1:DnoIG+QAMaF5NvxnGe/oKsgKcAc6PcUyl8q0VetfQ8s= gonum.org/v1/gonum v0.9.3/go.mod h1:TZumC3NeyVQskjXqmyWt4S3bINhy7B4eYwW69EbyX+0= +gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0 h1:OE9mWmgKkjJyEmDAAtGMPjXu+YNeGvK9VTSHY6+Qihc= gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0/go.mod h1:wa6Ws7BG/ESfp6dHfk7C6KdzKA7wR7u/rKwOGE66zvw= gonum.org/v1/plot v0.0.0-20190515093506-e2840ee46a6b/go.mod h1:Wt8AAjI+ypCyYX3nZBvf6cAIx93T+c/OS2HFAYskSZc= gonum.org/v1/plot v0.9.0/go.mod h1:3Pcqqmp6RHvJI72kgb8fThyUnav364FOsdDo2aGW5lY= google.golang.org/api v0.3.1/go.mod h1:6wY9I6uQWHQ8EM57III9mq/AjF+i8G65rmVagqKMtkk= -google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= -google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= -google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= -google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= -google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= -google.golang.org/api v0.14.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= -google.golang.org/api v0.15.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= -google.golang.org/api v0.17.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.18.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.20.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.22.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= 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.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= -google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= -google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= -google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= -google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= -google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= -google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190530194941-fb225487d101/go.mod h1:z3L6/3dTEVtUr6QSP8miRzeRqwQOioJ9I66odjN4I7s= -google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= -google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= -google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20200115191322-ca5a22157cba/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20200122232147-0452cf42e150/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20200204135345-fa8e72b47b90/go.mod h1:GmwEX6Z4W5gMy59cAlVYjN9JhxgbQH6Gn+gFDQe2lzA= -google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200224152610-e50cd9704f63/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= 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.20.0/go.mod h1:chYK+tFQF0nDUGJgXMSgLCQk3phJEuONr2DCgLDdAQM= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.0/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= -google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= google.golang.org/grpc v1.22.1/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= google.golang.org/grpc v1.23.1/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= -google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= -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= -gopkg.in/airbrake/gobrake.v2 v2.0.9/go.mod h1:/h5ZAUhDkGaJfjzjKLSjv6zCL6O0LLBxU4K+aSYdM/U= gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= @@ -754,13 +495,6 @@ gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qS gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= gopkg.in/gcfg.v1 v1.2.3/go.mod h1:yesOnuUOFQAhST5vPY4nbZsb/huCgGGXlipJsBn0b3o= -gopkg.in/gemnasium/logrus-airbrake-hook.v2 v2.1.2/go.mod h1:Xk6kEKp8OKb+X14hQBKWaSkCsqBpgog8nAV2xsGOxlo= -gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= -gopkg.in/jcmturner/aescts.v1 v1.0.1/go.mod h1:nsR8qBOg+OucoIW+WMhB3GspUQXq9XorLnQb9XtvcOo= -gopkg.in/jcmturner/dnsutils.v1 v1.0.1/go.mod h1:m3v+5svpVOhtFAP/wSz+yzh4Mc0Fg7eRhxkJMWSIz9Q= -gopkg.in/jcmturner/goidentity.v3 v3.0.0/go.mod h1:oG2kH0IvSYNIu80dVAyu/yoefjq1mNfM5bm88whjWx4= -gopkg.in/jcmturner/gokrb5.v7 v7.3.0/go.mod h1:l8VISx+WGYp+Fp7KRbsiUuXTTOnxIc3Tuvyavf11/WM= -gopkg.in/jcmturner/rpc.v1 v1.1.0/go.mod h1:YIdkC4XfD6GXbzje11McwsDuOlZQSb9W4vfLvuNnlv8= gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= @@ -770,21 +504,13 @@ gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw= -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-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= -honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= honnef.co/go/tools v0.1.3/go.mod h1:NgwopIslSNH47DimFoV78dnkksY2EFtX0ajyb3K/las= nhooyr.io/websocket v1.8.10 h1:mv4p+MnGrLDcPlBoWsvPP7XCzTYMXP9F9eIGoKbgx7Q= nhooyr.io/websocket v1.8.10/go.mod h1:rN9OFWIUwuxg4fR5tELlYC04bXYowCP9GX47ivo2l+c= -rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= -rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= -rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o= sourcegraph.com/sourcegraph/appdash v0.0.0-20190731080439-ebfcffb1b5c0/go.mod h1:hI742Nqp5OhwiqlzhgfbWU4mW4yO10fP+LoT9WOswdU= diff --git a/internal/worker/collector.go b/internal/worker/collector.go index d796a44..8fc9524 100644 --- a/internal/worker/collector.go +++ b/internal/worker/collector.go @@ -6,6 +6,9 @@ import ( "github.com/google/gopacket" "github.com/google/gopacket/pcap" "log" + "os" + "os/signal" + "syscall" ) type PacketSerializer interface { @@ -63,7 +66,7 @@ func NewCollector( // (signaling termination), it gracefully exits. // // Deferred actions ensure that the packet channel is closed and a log message is generated -// when the collector is closed. +// when the collector is closed func (c *Collector) Run(ctx context.Context) error { netInterface, err := c.netInterfaceProvider.GetNetworkInterface() if err != nil { @@ -74,18 +77,23 @@ func (c *Collector) Run(ctx context.Context) error { if err != nil { return fmt.Errorf("pcap open live: %w", err) } + defer handle.Close() if err := handle.SetBPFFilter(c.cfg.BPF); err != nil { return fmt.Errorf("set BPF filter: %w", err) } packetSource := gopacket.NewPacketSource(handle, handle.LinkType()) - packetsChan := make(chan gopacket.Packet) + signalChan := make(chan os.Signal, 1) + signal.Notify(signalChan, syscall.SIGINT, syscall.SIGTERM) + + ctx, cancel := context.WithCancel(ctx) + defer cancel() + defer func() { close(packetsChan) - log.Println("collector closed") }() @@ -95,17 +103,27 @@ func (c *Collector) Run(ctx context.Context) error { } }() - for { + go func() { select { - case packet, ok := <-packetSource.Packets(): - if !ok { - log.Printf("Error while reading packet: %v\n", err) - } + case <-signalChan: + log.Println("Received shutdown signal") + cancel() + case <-ctx.Done(): - packetsChan <- packet + } + }() + for { + select { case <-ctx.Done(): + log.Println("Shutting down gracefully") return nil + case packet := <-packetSource.Packets(): + if packet == nil { + log.Println("No more packets to read, shutting down") + return nil + } + packetsChan <- packet } } } From d4397db0eda24d7e3de5c4da33aa19847fd2bb07 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 10:29:15 +0200 Subject: [PATCH 12/27] Implemented graceful shutdown in server/server.go --- cmd/proxy/server/server.go | 37 +++++++++++++++++++++++++++++++----- internal/proxy/server.go | 6 ++++++ internal/worker/collector.go | 7 +++---- 3 files changed, 41 insertions(+), 9 deletions(-) diff --git a/cmd/proxy/server/server.go b/cmd/proxy/server/server.go index 5f4e8f1..b217e43 100644 --- a/cmd/proxy/server/server.go +++ b/cmd/proxy/server/server.go @@ -3,8 +3,13 @@ package server import ( "context" "embed" + "errors" "fmt" "log" + "net/http" + "os" + "os/signal" + "syscall" "time" "waffle/internal/visualize" @@ -75,11 +80,9 @@ func Run(proxyServerPort, visualizeServerPort string, yamlConfigBytes []byte, ce ) defender := guard.NewDefenseCoordinator([]guard.Defender{&guard.XSS{}}) - limiter := ratelimit.NewInMemoryLimiter(time.Minute * 5) visualizeServerPort = fmt.Sprintf(":%s", visualizeServerPort) - s := visualize.NewServer(visualizeServerPort) guardHandler := waf.NewHandler( @@ -90,13 +93,37 @@ func Run(proxyServerPort, visualizeServerPort string, yamlConfigBytes []byte, ce ) proxyServerPort = fmt.Sprintf(":%s", proxyServerPort) - proxyServer := proxy.NewServer(proxyServerPort, certificateProvider, guardHandler) log.Printf("Starting Waffle Proxy on port %s 🚀\n", proxyServerPort) - if err := proxyServer.Start(); err != nil { - log.Fatal(err.Error()) + signalChan := make(chan os.Signal, 1) + signal.Notify(signalChan, syscall.SIGINT, syscall.SIGTERM) + + ctx, cancel := context.WithCancel(context.Background()) + defer cancel() + + go func() { + if err := proxyServer.Start(); err != nil && !errors.Is(err, http.ErrServerClosed) { + log.Fatalf("Proxy server encountered an error: %v\n", err) + } + }() + + select { + case sig := <-signalChan: + log.Printf("Received shutdown signal: %s, shutting down gracefully...", sig) + cancel() + case <-ctx.Done(): + log.Println("Context canceled, shutting down...") + } + + shutdownCtx, shutdownCancel := context.WithTimeout(context.Background(), 10*time.Second) + defer shutdownCancel() + + if err := proxyServer.Shutdown(shutdownCtx); err != nil { + log.Printf("Error during server shutdown: %v", err) + } else { + log.Println("Proxy server shut down gracefully.") } return nil diff --git a/internal/proxy/server.go b/internal/proxy/server.go index b77f290..4ff2b1c 100644 --- a/internal/proxy/server.go +++ b/internal/proxy/server.go @@ -166,3 +166,9 @@ func (s *Server) Start() error { return nil } + +// Shutdown is a function which will be called when gracefully shutting down the server. +func (s *Server) Shutdown(ctx context.Context) error { + log.Println("Server shut down gracefully") + return nil +} diff --git a/internal/worker/collector.go b/internal/worker/collector.go index 8fc9524..b29dc3d 100644 --- a/internal/worker/collector.go +++ b/internal/worker/collector.go @@ -67,7 +67,7 @@ func NewCollector( // // Deferred actions ensure that the packet channel is closed and a log message is generated // when the collector is closed -func (c *Collector) Run(ctx context.Context) error { +func (c *Collector) Run() error { netInterface, err := c.netInterfaceProvider.GetNetworkInterface() if err != nil { return fmt.Errorf("get network interface using net interface provider: %w", err) @@ -89,12 +89,12 @@ func (c *Collector) Run(ctx context.Context) error { signalChan := make(chan os.Signal, 1) signal.Notify(signalChan, syscall.SIGINT, syscall.SIGTERM) - ctx, cancel := context.WithCancel(ctx) + ctx, cancel := context.WithCancel(context.Background()) defer cancel() defer func() { close(packetsChan) - log.Println("collector closed") + log.Println("Collector closed") }() go func() { @@ -109,7 +109,6 @@ func (c *Collector) Run(ctx context.Context) error { log.Println("Received shutdown signal") cancel() case <-ctx.Done(): - } }() From b9f1dc6271f6d65d906da6a19a1905a4710dbcdc Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 12:05:58 +0200 Subject: [PATCH 13/27] fix: Fixed CI/CD --- .github/workflows/go.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index d36ceed..b454d37 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -21,6 +21,9 @@ jobs: with: go-version: '1.22' + - name: Download go.mod dependencies + run: go mod download + - name: Build run: go build -v ./... From 7173f7e5fa50409cae9486fe582be2f4d6de9b66 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:15:56 +0200 Subject: [PATCH 14/27] fix: Repaired github action, and building docker image instead of project, since it can be later pushed --- .github/workflows/go.yml | 11 +++++++---- Taskfile.yaml | 11 +++++++---- 2 files changed, 14 insertions(+), 8 deletions(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index b454d37..355cc95 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -24,8 +24,11 @@ jobs: - name: Download go.mod dependencies run: go mod download - - name: Build - run: go build -v ./... - - - name: Test + - name: Test waffle WAF run: go test -v ./... + + - name: Install go-task to build docker image + run: sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -d -b ~/.local/bin + + - name: Build Waffle docker image + run: task dockerBuild \ No newline at end of file diff --git a/Taskfile.yaml b/Taskfile.yaml index a53b617..545108f 100644 --- a/Taskfile.yaml +++ b/Taskfile.yaml @@ -1,5 +1,8 @@ version: '3' +env: + IMAGE_NAME: '{{ .IMAGE_NAME | default "qniw984/waffle:1.0.0" }}' + tasks: certs_windows: cmds: @@ -27,10 +30,10 @@ tasks: cmds: - mockery - docker-build: + dockerBuild: cmds: - - docker build -t ${IMAGE_NAME} -f .\build\Dockerfile . + - docker build -t {{ .IMAGE_NAME }] -f .\build\Dockerfile . - docker-push: + dockerPush: cmds: - - docker push ${IMAGE_NAME} + - docker push {{ .IMAGE_NAME }] From 9dc854acec9969a3fa32b20cc5a26549c7964be7 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:17:18 +0200 Subject: [PATCH 15/27] fix: Commented unused code --- pkg/permission/windows.go | 56 +++++++++++++++++++-------------------- 1 file changed, 28 insertions(+), 28 deletions(-) diff --git a/pkg/permission/windows.go b/pkg/permission/windows.go index 2169951..00cf5cf 100644 --- a/pkg/permission/windows.go +++ b/pkg/permission/windows.go @@ -1,30 +1,30 @@ package permission -import ( - "fmt" - "golang.org/x/sys/windows" - "os" - "strings" - "syscall" -) - -func RunMeElevated() error { - verb := "runas" - exe, _ := os.Executable() - cwd, _ := os.Getwd() - args := strings.Join(os.Args[1:], " ") - - verbPtr, _ := syscall.UTF16PtrFromString(verb) - exePtr, _ := syscall.UTF16PtrFromString(exe) - cwdPtr, _ := syscall.UTF16PtrFromString(cwd) - argPtr, _ := syscall.UTF16PtrFromString(args) - - var showCmd int32 = 1 //SW_NORMAL - - err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) - if err != nil { - return fmt.Errorf("windows shell execute: %w", err) - } - - return nil -} +//import ( +// "fmt" +// "golang.org/x/sys/windows" +// "os" +// "strings" +// "syscall" +//) +// +//func RunMeElevated() error { +// verb := "runas" +// exe, _ := os.Executable() +// cwd, _ := os.Getwd() +// args := strings.Join(os.Args[1:], " ") +// +// verbPtr, _ := syscall.UTF16PtrFromString(verb) +// exePtr, _ := syscall.UTF16PtrFromString(exe) +// cwdPtr, _ := syscall.UTF16PtrFromString(cwd) +// argPtr, _ := syscall.UTF16PtrFromString(args) +// +// var showCmd int32 = 1 //SW_NORMAL +// +// err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) +// if err != nil { +// return fmt.Errorf("windows shell execute: %w", err) +// } +// +// return nil +//} From ea9ac7dbeed51db0637121ddc12fe356c4ee3a6f Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:19:43 +0200 Subject: [PATCH 16/27] Running tests only in internal package instead of ./... --- .github/workflows/go.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index 355cc95..913f4e8 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -25,7 +25,7 @@ jobs: run: go mod download - name: Test waffle WAF - run: go test -v ./... + run: go test -v ./internal - name: Install go-task to build docker image run: sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -d -b ~/.local/bin From df6733f6e24237fcd37b8ce2cf46c4af43d5a893 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:20:04 +0200 Subject: [PATCH 17/27] fix: Running tests in every folder in internal pkg --- .github/workflows/go.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index 913f4e8..889d397 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -25,7 +25,7 @@ jobs: run: go mod download - name: Test waffle WAF - run: go test -v ./internal + run: go test -v ./internal/... - name: Install go-task to build docker image run: sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -d -b ~/.local/bin From f57c01dfc9d541584d97eeafab5d0437ff24099b Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:21:36 +0200 Subject: [PATCH 18/27] fix: Repairing Github Pipeline; Added pcap installation --- .github/workflows/go.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index 889d397..14b2383 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -21,6 +21,9 @@ jobs: with: go-version: '1.22' + - name: Install pcap deps + run: apt install -y libpcap-dev + - name: Download go.mod dependencies run: go mod download From b6c965d224e4d4b72bd98364c66281a1b1d9d453 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:22:26 +0200 Subject: [PATCH 19/27] fix: Repairing Github Pipeline; Added sudo to pcap install --- .github/workflows/go.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index 14b2383..c636c38 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -22,7 +22,7 @@ jobs: go-version: '1.22' - name: Install pcap deps - run: apt install -y libpcap-dev + run: sudo apt install -y libpcap-dev - name: Download go.mod dependencies run: go mod download From 7a79fa6b8c7be783756903522504be78ae20beb5 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Tue, 22 Oct 2024 15:25:02 +0200 Subject: [PATCH 20/27] Testing everything in waffle folder, since pcap error is resolved. --- .github/workflows/go.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index c636c38..f32a8aa 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -28,7 +28,7 @@ jobs: run: go mod download - name: Test waffle WAF - run: go test -v ./internal/... + run: go test -v ./... - name: Install go-task to build docker image run: sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -d -b ~/.local/bin From 0ebb7d336e432443f972a2743c0d9a7da000c642 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 11:53:33 +0200 Subject: [PATCH 21/27] changed username in docker image to YOUR_USERNAME --- Taskfile.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Taskfile.yaml b/Taskfile.yaml index 545108f..9a97fb0 100644 --- a/Taskfile.yaml +++ b/Taskfile.yaml @@ -1,7 +1,7 @@ version: '3' env: - IMAGE_NAME: '{{ .IMAGE_NAME | default "qniw984/waffle:1.0.0" }}' + IMAGE_NAME: '{{ .IMAGE_NAME | default "/:>" }}' tasks: certs_windows: From 44cc1e925b991453b8d9d2c8ffdbf2d39b93ae26 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 11:54:05 +0200 Subject: [PATCH 22/27] Removed unused packages from project. --- internal/rule/compiler.go | 81 ----- internal/rule/compiler_test.go | 96 ------ internal/rule/expression_tree.go | 44 --- internal/rule/expression_tree_test.go | 118 ------- internal/rule/factory.go | 154 --------- internal/rule/factory_test.go | 158 ---------- internal/rule/mock_Builder.go | 92 ------ internal/rule/mock_ExpressionTreeBuilder.go | 91 ------ internal/rule/mock_ExpressionTreeFactory.go | 90 ------ internal/rule/mock_Tokenizer.go | 91 ------ internal/rule/mock_node.go | 82 ----- internal/rule/node.go | 55 ---- internal/rule/node_test.go | 328 -------------------- internal/rule/predicate.go | 44 --- internal/rule/predicate_test.go | 81 ----- internal/rule/tokenizer.go | 201 ------------ internal/rule/tokenizer_test.go | 144 --------- pkg/permission/windows.go | 30 -- 18 files changed, 1980 deletions(-) delete mode 100644 internal/rule/compiler.go delete mode 100644 internal/rule/compiler_test.go delete mode 100644 internal/rule/expression_tree.go delete mode 100644 internal/rule/expression_tree_test.go delete mode 100644 internal/rule/factory.go delete mode 100644 internal/rule/factory_test.go delete mode 100644 internal/rule/mock_Builder.go delete mode 100644 internal/rule/mock_ExpressionTreeBuilder.go delete mode 100644 internal/rule/mock_ExpressionTreeFactory.go delete mode 100644 internal/rule/mock_Tokenizer.go delete mode 100644 internal/rule/mock_node.go delete mode 100644 internal/rule/node.go delete mode 100644 internal/rule/node_test.go delete mode 100644 internal/rule/predicate.go delete mode 100644 internal/rule/predicate_test.go delete mode 100644 internal/rule/tokenizer.go delete mode 100644 internal/rule/tokenizer_test.go delete mode 100644 pkg/permission/windows.go diff --git a/internal/rule/compiler.go b/internal/rule/compiler.go deleted file mode 100644 index 952a581..0000000 --- a/internal/rule/compiler.go +++ /dev/null @@ -1,81 +0,0 @@ -package rule - -import ( - "errors" - "fmt" - "strings" -) - -const ( - variableExpressionDivider = "=>" -) - -var ( - ErrEmptyPredicateName = errors.New("predicate name is empty") - ErrEmptyPredicateValue = errors.New("value to compile predicate is empty") - ErrMissingPredicateParts = errors.New("predicate parts are missing") - ErrInvalidPredicate = errors.New("predicate is invalid") -) - -type Compiler interface { - // Compile compiles rule to the functional predicate. - // Name - is a name of the predicate. - // Value - is a value that should be compiled into predicate. - Compile(name, value string) (*Predicate, error) -} - -type CustomCompiler struct { - builder Builder -} - -var _ Compiler = (*CustomCompiler)(nil) - -func NewCustomCompiler() *CustomCompiler { - return &CustomCompiler{ - builder: newPredicateBuilder(), - } -} - -func (c *CustomCompiler) Compile(name, value string) (*Predicate, error) { - if err := validateInput(name, value); err != nil { - return nil, fmt.Errorf("validate name and value: %w", err) - } - - variable, expression, err := getVariableAndLogicalExpression(value) - if err != nil { - return nil, fmt.Errorf("get variable and logical expression for predicate: %w", err) - } - - pred, err := c.builder.Build(name, variable, expression) - if err != nil { - return nil, fmt.Errorf("build predicate: %w", err) - } - - return pred, nil -} - -func validateInput(name string, value string) error { - if len(name) == 0 { - return ErrEmptyPredicateName - } - - if len(value) == 0 { - return ErrEmptyPredicateValue - } - - return nil -} - -func getVariableAndLogicalExpression(value string) (string, string, error) { - values := strings.Split(value, variableExpressionDivider) - - if len(values) < 2 { - return "", "", ErrMissingPredicateParts - } - - if len(values) > 2 { - return "", "", ErrInvalidPredicate - } - - return values[0], values[1], nil -} diff --git a/internal/rule/compiler_test.go b/internal/rule/compiler_test.go deleted file mode 100644 index 058c466..0000000 --- a/internal/rule/compiler_test.go +++ /dev/null @@ -1,96 +0,0 @@ -package rule - -import ( - "errors" - "github.com/stretchr/testify/mock" - "reflect" - "testing" -) - -func TestCustomCompiler_Compile(t *testing.T) { - type fields struct { - builderFunc func() Builder - } - type args struct { - name string - value string - } - tests := []struct { - name string - fields fields - args args - want *Predicate - wantErr bool - }{ - { - name: "empty name, error returned", - args: args{ - name: "", - value: "p => p", - }, - wantErr: true, - }, - { - name: "empty value, error returned", - args: args{ - name: "name", - value: "", - }, - wantErr: true, - }, - { - name: "missing => sign, error returned", - args: args{ - name: "name", - value: "p p.length() > 0", - }, - wantErr: true, - }, - { - name: "to many => signs, error returned", - args: args{ - name: "name", - value: "p => p.format() => json", - }, - wantErr: true, - }, - { - name: "builder returns error, error returned", - fields: fields{ - builderFunc: func() Builder { - b := NewMockBuilder(t) - - b.EXPECT().Build(mock.Anything, mock.Anything, mock.Anything). - Return(nil, errors.New("err")) - - return b - }, - }, - args: args{ - name: "name", - value: "p => p.length() > 0", - }, - wantErr: true, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var builder Builder - if tt.fields.builderFunc != nil { - builder = tt.fields.builderFunc() - } - - c := &CustomCompiler{ - builder: builder, - } - got, err := c.Compile(tt.args.name, tt.args.value) - if (err != nil) != tt.wantErr { - t.Errorf("Compile() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("Compile() got = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/internal/rule/expression_tree.go b/internal/rule/expression_tree.go deleted file mode 100644 index e8837f4..0000000 --- a/internal/rule/expression_tree.go +++ /dev/null @@ -1,44 +0,0 @@ -package rule - -import "fmt" - -type expressionTree node - -type ExpressionTreeFactory interface { - CreateExpressionTree(tokens []Token) (expressionTree, error) -} - -// Tokenizer should be implemented by the structs that tokenizes input. -type Tokenizer interface { - // BuildTokens builds tokens based on the variable and expression. - // Returns error if tokenizer cannot be done with the input. - BuildTokens(variable string, expression string) ([]Token, error) -} - -type expressionTreeBuilder struct { - tokenizer Tokenizer - expressionTreeFactory ExpressionTreeFactory -} - -var _ ExpressionTreeBuilder = (*expressionTreeBuilder)(nil) - -func newExpressionTreeBuilder() *expressionTreeBuilder { - return &expressionTreeBuilder{ - tokenizer: &tokenizer{}, - expressionTreeFactory: &expressionTreeFactory{}, - } -} - -func (c *expressionTreeBuilder) BuildExpressionTree(variable, expression string) (expressionTree, error) { - tokens, err := c.tokenizer.BuildTokens(variable, expression) - if err != nil { - return nil, fmt.Errorf("build tokens: %w", err) - } - - tree, err := c.expressionTreeFactory.CreateExpressionTree(tokens) - if err != nil { - return nil, fmt.Errorf("create expression tree: %w", err) - } - - return tree, nil -} diff --git a/internal/rule/expression_tree_test.go b/internal/rule/expression_tree_test.go deleted file mode 100644 index d19e102..0000000 --- a/internal/rule/expression_tree_test.go +++ /dev/null @@ -1,118 +0,0 @@ -package rule - -import ( - "errors" - "reflect" - "testing" - - "github.com/stretchr/testify/mock" -) - -func Test_expressionTreeBuilder_BuildExpressionTree(t *testing.T) { - type fields struct { - tokenizerFunc func() Tokenizer - expressionTreeFactoryFunc func() ExpressionTreeFactory - } - type args struct { - variable string - expression string - } - tests := []struct { - name string - fields fields - args args - want expressionTree - wantErr bool - }{ - { - name: "tokenizer returns error, error returned", - fields: fields{ - tokenizerFunc: func() Tokenizer { - tkn := NewMockTokenizer(t) - - tkn.EXPECT(). - BuildTokens(mock.Anything, mock.Anything). - Return(nil, errors.New("error")) - - return tkn - }, - }, - wantErr: true, - }, - { - name: "expression tree factory returns error, error returned", - fields: fields{ - tokenizerFunc: func() Tokenizer { - tkn := NewMockTokenizer(t) - - tkn.EXPECT(). - BuildTokens(mock.Anything, mock.Anything). - Return(testTokens, nil) - - return tkn - }, - expressionTreeFactoryFunc: func() ExpressionTreeFactory { - etf := NewMockExpressionTreeFactory(t) - - etf.EXPECT(). - CreateExpressionTree(mock.Anything). - Return(nil, errors.New("error")) - - return etf - }, - }, - wantErr: true, - }, - { - name: "expression tree factory returns expression tree, tree returned", - fields: fields{ - tokenizerFunc: func() Tokenizer { - tkn := NewMockTokenizer(t) - - tkn.EXPECT(). - BuildTokens(mock.Anything, mock.Anything). - Return(testTokens, nil) - - return tkn - }, - expressionTreeFactoryFunc: func() ExpressionTreeFactory { - etf := NewMockExpressionTreeFactory(t) - - etf.EXPECT(). - CreateExpressionTree(testTokens). - Return(and{}, nil) - - return etf - }, - }, - wantErr: false, - want: and{}, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var expressionTreeFactory ExpressionTreeFactory - if tt.fields.expressionTreeFactoryFunc != nil { - expressionTreeFactory = tt.fields.expressionTreeFactoryFunc() - } - - var tokenizer Tokenizer - if tt.fields.tokenizerFunc != nil { - tokenizer = tt.fields.tokenizerFunc() - } - - c := &expressionTreeBuilder{ - tokenizer: tokenizer, - expressionTreeFactory: expressionTreeFactory, - } - got, err := c.BuildExpressionTree(tt.args.variable, tt.args.expression) - if (err != nil) != tt.wantErr { - t.Errorf("BuildExpressionTree() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("BuildExpressionTree() got = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/internal/rule/factory.go b/internal/rule/factory.go deleted file mode 100644 index aac6569..0000000 --- a/internal/rule/factory.go +++ /dev/null @@ -1,154 +0,0 @@ -package rule - -import "github.com/emirpasic/gods/stacks/arraystack" - -var ( - // operatorPrecedence is a slice of operators in the precedence order. - // The most important ones are on the top, the lest important ones are on the bottom. - // Also if there are two operators: A and B, and if those are in the same row, - // those are equal in terms of precedence. - operatorPrecedence = [][]string{ - // Postfix - { - tokenLParen, - tokenRParen, - tokenDot, - }, - // Relational - { - tokenMoreThan, - tokenLessThan, - }, - // Equality - { - tokenEqual, - tokenNotEqual, - }, - // AND - { - tokenDoubleAmpersand, - }, - // OR - { - tokenOr, - }, - // Dot - { - tokenDot, - }, - } -) - -type expressionTreeFactory struct { -} - -var _ ExpressionTreeFactory = (*expressionTreeFactory)(nil) - -func (e *expressionTreeFactory) CreateExpressionTree(tokens []Token) (expressionTree, error) { - return nil, nil -} - -// 1. While there are tokens to be read: -// 2. Read a token -// 3. If it's a number add it to queue -// 4. If it's an operator -// 5. While there's an operator on the top of the stack with greater precedence: -// 6. Pop operators from the stack onto the output queue -// 7. Push the current operator onto the stack -// 8. If it's a left bracket push it onto the stack -// 9. If it's a right bracket -// 10. While there's not a left bracket at the top of the stack: -// 11. Pop operators from the stack onto the output queue. -// 12. Pop the left bracket from the stack and discard it -// 13. While there are operators on the stack, pop them to the queue - -func reversePolishNotationSort(tokens []Token) []Token { - - var outputTokens []Token - operatorStack := arraystack.New() - - for _, token := range tokens { - if isOperator(token) { - if v, ok := operatorStack.Peek(); ok { - vt := v.(Token) - if isTokenWithGreaterPrecedenceFromThePrecedenceSlice(token, vt) { - for !operatorStack.Empty() { - v, ok := operatorStack.Pop() - if !ok { - continue - } - - vt := v.(Token) - - outputTokens = append(outputTokens, vt) - } - - operatorStack.Push(token) - } - } - - continue - } - - if token.Name == tokenLParen { - operatorStack.Push(token) - - continue - } - - if token.Name == tokenRParen { - var lParen *Token - for !operatorStack.Empty() || lParen == nil { - v, ok := operatorStack.Pop() - if !ok { - continue - } - - vt := v.(Token) - - if vt.Name == tokenLParen { - lParen = &vt - continue - } - - outputTokens = append(outputTokens, vt) - } - - continue - } - - tokens = append(tokens, token) - } - - return tokens -} - -func isOperator(tkn Token) bool { - for _, tknOperator := range tokensOperators { - if tkn.Name == tknOperator { - return true - } - } - - return false -} - -func isTokenWithGreaterPrecedenceFromThePrecedenceSlice(tkn Token, lastTknFromStack Token) bool { - var ( - tknIdx int - lastTknFromStackIdx int - ) - - for i, op := range operatorPrecedence { - for _, opp := range op { - if opp == tkn.Name { - tknIdx = i - } - if opp == lastTknFromStack.Name { - lastTknFromStackIdx = i - } - } - } - - return lastTknFromStackIdx < tknIdx -} diff --git a/internal/rule/factory_test.go b/internal/rule/factory_test.go deleted file mode 100644 index df281e3..0000000 --- a/internal/rule/factory_test.go +++ /dev/null @@ -1,158 +0,0 @@ -package rule - -import ( - "reflect" - "testing" -) - -func Test_isOperator(t *testing.T) { - type args struct { - tkn Token - } - tests := []struct { - name string - args args - want bool - }{ - { - name: "is an operator, true returned", - args: args{ - tkn: Token{ - Name: tokenMoreThan, - }, - }, - want: true, - }, - { - name: "isn't an operator, false returned", - args: args{ - tkn: Token{ - Name: tokenLParen, - }, - }, - want: false, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := isOperator(tt.args.tkn); got != tt.want { - t.Errorf("isOperator() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_isTokenWithGreaterPrecedenceFromThePrecedenceSlices(t *testing.T) { - type args struct { - tkn Token - lastTknFromStack Token - } - tests := []struct { - name string - args args - want bool - }{ - { - name: "token with greater precedence, true returned", - args: args{ - tkn: Token{ - Name: tokenOr, - Value: tokenOr, - }, - lastTknFromStack: Token{ - Name: tokenLParen, - Value: tokenLParen, - }, - }, - want: true, - }, - { - name: "token with less precedence, false returned", - args: args{ - tkn: Token{ - Name: tokenLParen, - Value: tokenLParen, - }, - lastTknFromStack: Token{ - Name: tokenOr, - Value: tokenOr, - }, - }, - want: false, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := isTokenWithGreaterPrecedenceFromThePrecedenceSlice(tt.args.tkn, tt.args.lastTknFromStack); got != tt.want { - t.Errorf("isTokenWithGreaterPrecedenceFromThePrecedenceSlice() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_reversePolishNotationSort(t *testing.T) { - type args struct { - tokens []Token - } - tests := []struct { - name string - args args - want []Token - }{ - { - name: "tokens should be ordered", - args: args{ - tokens: []Token{ - { - Name: tokenFunction, - Value: "LEN", - }, - { - Name: tokenLParen, - Value: tokenLParen, - }, - { - Name: tokenVariable, - Value: "p", - }, - { - Name: tokenDot, - Value: ".", - }, - { - Name: tokenField, - Value: "headers", - }, - { - Name: tokenRParen, - Value: tokenRParen, - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenMoreThan, - Value: tokenMoreThan, - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenNumber, - Value: "5", - }, - }, - }, - want: nil, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := reversePolishNotationSort(tt.args.tokens); !reflect.DeepEqual(got, tt.want) { - t.Errorf("reversePolishNotationSort() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/internal/rule/mock_Builder.go b/internal/rule/mock_Builder.go deleted file mode 100644 index d69d712..0000000 --- a/internal/rule/mock_Builder.go +++ /dev/null @@ -1,92 +0,0 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. - -package rule - -import mock "github.com/stretchr/testify/mock" - -// MockBuilder is an autogenerated mock type for the Builder type -type MockBuilder struct { - mock.Mock -} - -type MockBuilder_Expecter struct { - mock *mock.Mock -} - -func (_m *MockBuilder) EXPECT() *MockBuilder_Expecter { - return &MockBuilder_Expecter{mock: &_m.Mock} -} - -// Build provides a mock function with given fields: name, variable, expression -func (_m *MockBuilder) Build(name string, variable string, expression string) (*Predicate, error) { - ret := _m.Called(name, variable, expression) - - if len(ret) == 0 { - panic("no return value specified for Build") - } - - var r0 *Predicate - var r1 error - if rf, ok := ret.Get(0).(func(string, string, string) (*Predicate, error)); ok { - return rf(name, variable, expression) - } - if rf, ok := ret.Get(0).(func(string, string, string) *Predicate); ok { - r0 = rf(name, variable, expression) - } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).(*Predicate) - } - } - - if rf, ok := ret.Get(1).(func(string, string, string) error); ok { - r1 = rf(name, variable, expression) - } else { - r1 = ret.Error(1) - } - - return r0, r1 -} - -// MockBuilder_Build_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Build' -type MockBuilder_Build_Call struct { - *mock.Call -} - -// Build is a helper method to define mock.On call -// - name string -// - variable string -// - expression string -func (_e *MockBuilder_Expecter) Build(name interface{}, variable interface{}, expression interface{}) *MockBuilder_Build_Call { - return &MockBuilder_Build_Call{Call: _e.mock.On("Build", name, variable, expression)} -} - -func (_c *MockBuilder_Build_Call) Run(run func(name string, variable string, expression string)) *MockBuilder_Build_Call { - _c.Call.Run(func(args mock.Arguments) { - run(args[0].(string), args[1].(string), args[2].(string)) - }) - return _c -} - -func (_c *MockBuilder_Build_Call) Return(_a0 *Predicate, _a1 error) *MockBuilder_Build_Call { - _c.Call.Return(_a0, _a1) - return _c -} - -func (_c *MockBuilder_Build_Call) RunAndReturn(run func(string, string, string) (*Predicate, error)) *MockBuilder_Build_Call { - _c.Call.Return(run) - return _c -} - -// NewMockBuilder creates a new instance of MockBuilder. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. -// The first argument is typically a *testing.T value. -func NewMockBuilder(t interface { - mock.TestingT - Cleanup(func()) -}) *MockBuilder { - mock := &MockBuilder{} - mock.Mock.Test(t) - - t.Cleanup(func() { mock.AssertExpectations(t) }) - - return mock -} diff --git a/internal/rule/mock_ExpressionTreeBuilder.go b/internal/rule/mock_ExpressionTreeBuilder.go deleted file mode 100644 index 64b31f7..0000000 --- a/internal/rule/mock_ExpressionTreeBuilder.go +++ /dev/null @@ -1,91 +0,0 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. - -package rule - -import mock "github.com/stretchr/testify/mock" - -// MockExpressionTreeBuilder is an autogenerated mock type for the ExpressionTreeBuilder type -type MockExpressionTreeBuilder struct { - mock.Mock -} - -type MockExpressionTreeBuilder_Expecter struct { - mock *mock.Mock -} - -func (_m *MockExpressionTreeBuilder) EXPECT() *MockExpressionTreeBuilder_Expecter { - return &MockExpressionTreeBuilder_Expecter{mock: &_m.Mock} -} - -// BuildExpressionTree provides a mock function with given fields: variable, expression -func (_m *MockExpressionTreeBuilder) BuildExpressionTree(variable string, expression string) (expressionTree, error) { - ret := _m.Called(variable, expression) - - if len(ret) == 0 { - panic("no return value specified for BuildExpressionTree") - } - - var r0 expressionTree - var r1 error - if rf, ok := ret.Get(0).(func(string, string) (expressionTree, error)); ok { - return rf(variable, expression) - } - if rf, ok := ret.Get(0).(func(string, string) expressionTree); ok { - r0 = rf(variable, expression) - } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).(expressionTree) - } - } - - if rf, ok := ret.Get(1).(func(string, string) error); ok { - r1 = rf(variable, expression) - } else { - r1 = ret.Error(1) - } - - return r0, r1 -} - -// MockExpressionTreeBuilder_BuildExpressionTree_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'BuildExpressionTree' -type MockExpressionTreeBuilder_BuildExpressionTree_Call struct { - *mock.Call -} - -// BuildExpressionTree is a helper method to define mock.On call -// - variable string -// - expression string -func (_e *MockExpressionTreeBuilder_Expecter) BuildExpressionTree(variable interface{}, expression interface{}) *MockExpressionTreeBuilder_BuildExpressionTree_Call { - return &MockExpressionTreeBuilder_BuildExpressionTree_Call{Call: _e.mock.On("BuildExpressionTree", variable, expression)} -} - -func (_c *MockExpressionTreeBuilder_BuildExpressionTree_Call) Run(run func(variable string, expression string)) *MockExpressionTreeBuilder_BuildExpressionTree_Call { - _c.Call.Run(func(args mock.Arguments) { - run(args[0].(string), args[1].(string)) - }) - return _c -} - -func (_c *MockExpressionTreeBuilder_BuildExpressionTree_Call) Return(_a0 expressionTree, _a1 error) *MockExpressionTreeBuilder_BuildExpressionTree_Call { - _c.Call.Return(_a0, _a1) - return _c -} - -func (_c *MockExpressionTreeBuilder_BuildExpressionTree_Call) RunAndReturn(run func(string, string) (expressionTree, error)) *MockExpressionTreeBuilder_BuildExpressionTree_Call { - _c.Call.Return(run) - return _c -} - -// NewMockExpressionTreeBuilder creates a new instance of MockExpressionTreeBuilder. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. -// The first argument is typically a *testing.T value. -func NewMockExpressionTreeBuilder(t interface { - mock.TestingT - Cleanup(func()) -}) *MockExpressionTreeBuilder { - mock := &MockExpressionTreeBuilder{} - mock.Mock.Test(t) - - t.Cleanup(func() { mock.AssertExpectations(t) }) - - return mock -} diff --git a/internal/rule/mock_ExpressionTreeFactory.go b/internal/rule/mock_ExpressionTreeFactory.go deleted file mode 100644 index 667119f..0000000 --- a/internal/rule/mock_ExpressionTreeFactory.go +++ /dev/null @@ -1,90 +0,0 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. - -package rule - -import mock "github.com/stretchr/testify/mock" - -// MockExpressionTreeFactory is an autogenerated mock type for the ExpressionTreeFactory type -type MockExpressionTreeFactory struct { - mock.Mock -} - -type MockExpressionTreeFactory_Expecter struct { - mock *mock.Mock -} - -func (_m *MockExpressionTreeFactory) EXPECT() *MockExpressionTreeFactory_Expecter { - return &MockExpressionTreeFactory_Expecter{mock: &_m.Mock} -} - -// CreateExpressionTree provides a mock function with given fields: tokens -func (_m *MockExpressionTreeFactory) CreateExpressionTree(tokens []Token) (expressionTree, error) { - ret := _m.Called(tokens) - - if len(ret) == 0 { - panic("no return value specified for CreateExpressionTree") - } - - var r0 expressionTree - var r1 error - if rf, ok := ret.Get(0).(func([]Token) (expressionTree, error)); ok { - return rf(tokens) - } - if rf, ok := ret.Get(0).(func([]Token) expressionTree); ok { - r0 = rf(tokens) - } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).(expressionTree) - } - } - - if rf, ok := ret.Get(1).(func([]Token) error); ok { - r1 = rf(tokens) - } else { - r1 = ret.Error(1) - } - - return r0, r1 -} - -// MockExpressionTreeFactory_CreateExpressionTree_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'CreateExpressionTree' -type MockExpressionTreeFactory_CreateExpressionTree_Call struct { - *mock.Call -} - -// CreateExpressionTree is a helper method to define mock.On call -// - tokens []Token -func (_e *MockExpressionTreeFactory_Expecter) CreateExpressionTree(tokens interface{}) *MockExpressionTreeFactory_CreateExpressionTree_Call { - return &MockExpressionTreeFactory_CreateExpressionTree_Call{Call: _e.mock.On("CreateExpressionTree", tokens)} -} - -func (_c *MockExpressionTreeFactory_CreateExpressionTree_Call) Run(run func(tokens []Token)) *MockExpressionTreeFactory_CreateExpressionTree_Call { - _c.Call.Run(func(args mock.Arguments) { - run(args[0].([]Token)) - }) - return _c -} - -func (_c *MockExpressionTreeFactory_CreateExpressionTree_Call) Return(_a0 expressionTree, _a1 error) *MockExpressionTreeFactory_CreateExpressionTree_Call { - _c.Call.Return(_a0, _a1) - return _c -} - -func (_c *MockExpressionTreeFactory_CreateExpressionTree_Call) RunAndReturn(run func([]Token) (expressionTree, error)) *MockExpressionTreeFactory_CreateExpressionTree_Call { - _c.Call.Return(run) - return _c -} - -// NewMockExpressionTreeFactory creates a new instance of MockExpressionTreeFactory. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. -// The first argument is typically a *testing.T value. -func NewMockExpressionTreeFactory(t interface { - mock.TestingT - Cleanup(func()) -}) *MockExpressionTreeFactory { - mock := &MockExpressionTreeFactory{} - mock.Mock.Test(t) - - t.Cleanup(func() { mock.AssertExpectations(t) }) - - return mock -} diff --git a/internal/rule/mock_Tokenizer.go b/internal/rule/mock_Tokenizer.go deleted file mode 100644 index ada4866..0000000 --- a/internal/rule/mock_Tokenizer.go +++ /dev/null @@ -1,91 +0,0 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. - -package rule - -import mock "github.com/stretchr/testify/mock" - -// MockTokenizer is an autogenerated mock type for the Tokenizer type -type MockTokenizer struct { - mock.Mock -} - -type MockTokenizer_Expecter struct { - mock *mock.Mock -} - -func (_m *MockTokenizer) EXPECT() *MockTokenizer_Expecter { - return &MockTokenizer_Expecter{mock: &_m.Mock} -} - -// BuildTokens provides a mock function with given fields: variable, expression -func (_m *MockTokenizer) BuildTokens(variable string, expression string) ([]Token, error) { - ret := _m.Called(variable, expression) - - if len(ret) == 0 { - panic("no return value specified for BuildTokens") - } - - var r0 []Token - var r1 error - if rf, ok := ret.Get(0).(func(string, string) ([]Token, error)); ok { - return rf(variable, expression) - } - if rf, ok := ret.Get(0).(func(string, string) []Token); ok { - r0 = rf(variable, expression) - } else { - if ret.Get(0) != nil { - r0 = ret.Get(0).([]Token) - } - } - - if rf, ok := ret.Get(1).(func(string, string) error); ok { - r1 = rf(variable, expression) - } else { - r1 = ret.Error(1) - } - - return r0, r1 -} - -// MockTokenizer_BuildTokens_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'BuildTokens' -type MockTokenizer_BuildTokens_Call struct { - *mock.Call -} - -// BuildTokens is a helper method to define mock.On call -// - variable string -// - expression string -func (_e *MockTokenizer_Expecter) BuildTokens(variable interface{}, expression interface{}) *MockTokenizer_BuildTokens_Call { - return &MockTokenizer_BuildTokens_Call{Call: _e.mock.On("BuildTokens", variable, expression)} -} - -func (_c *MockTokenizer_BuildTokens_Call) Run(run func(variable string, expression string)) *MockTokenizer_BuildTokens_Call { - _c.Call.Run(func(args mock.Arguments) { - run(args[0].(string), args[1].(string)) - }) - return _c -} - -func (_c *MockTokenizer_BuildTokens_Call) Return(_a0 []Token, _a1 error) *MockTokenizer_BuildTokens_Call { - _c.Call.Return(_a0, _a1) - return _c -} - -func (_c *MockTokenizer_BuildTokens_Call) RunAndReturn(run func(string, string) ([]Token, error)) *MockTokenizer_BuildTokens_Call { - _c.Call.Return(run) - return _c -} - -// NewMockTokenizer creates a new instance of MockTokenizer. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. -// The first argument is typically a *testing.T value. -func NewMockTokenizer(t interface { - mock.TestingT - Cleanup(func()) -}) *MockTokenizer { - mock := &MockTokenizer{} - mock.Mock.Test(t) - - t.Cleanup(func() { mock.AssertExpectations(t) }) - - return mock -} diff --git a/internal/rule/mock_node.go b/internal/rule/mock_node.go deleted file mode 100644 index d08b928..0000000 --- a/internal/rule/mock_node.go +++ /dev/null @@ -1,82 +0,0 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. - -package rule - -import ( - request "waffle/internal/request" - - mock "github.com/stretchr/testify/mock" -) - -// Mocknode is an autogenerated mock type for the node type -type Mocknode struct { - mock.Mock -} - -type Mocknode_Expecter struct { - mock *mock.Mock -} - -func (_m *Mocknode) EXPECT() *Mocknode_Expecter { - return &Mocknode_Expecter{mock: &_m.Mock} -} - -// Eval provides a mock function with given fields: r -func (_m *Mocknode) Eval(r request.Wrapper) bool { - ret := _m.Called(r) - - if len(ret) == 0 { - panic("no return value specified for Eval") - } - - var r0 bool - if rf, ok := ret.Get(0).(func(request.Wrapper) bool); ok { - r0 = rf(r) - } else { - r0 = ret.Get(0).(bool) - } - - return r0 -} - -// Mocknode_Eval_Call is a *mock.Call that shadows Run/Return methods with type explicit version for method 'Eval' -type Mocknode_Eval_Call struct { - *mock.Call -} - -// Eval is a helper method to define mock.On call -// - r request.Wrapper -func (_e *Mocknode_Expecter) Eval(r interface{}) *Mocknode_Eval_Call { - return &Mocknode_Eval_Call{Call: _e.mock.On("Eval", r)} -} - -func (_c *Mocknode_Eval_Call) Run(run func(r request.Wrapper)) *Mocknode_Eval_Call { - _c.Call.Run(func(args mock.Arguments) { - run(args[0].(request.Wrapper)) - }) - return _c -} - -func (_c *Mocknode_Eval_Call) Return(_a0 bool) *Mocknode_Eval_Call { - _c.Call.Return(_a0) - return _c -} - -func (_c *Mocknode_Eval_Call) RunAndReturn(run func(request.Wrapper) bool) *Mocknode_Eval_Call { - _c.Call.Return(run) - return _c -} - -// NewMocknode creates a new instance of Mocknode. It also registers a testing interface on the mock and a cleanup function to assert the mocks expectations. -// The first argument is typically a *testing.T value. -func NewMocknode(t interface { - mock.TestingT - Cleanup(func()) -}) *Mocknode { - mock := &Mocknode{} - mock.Mock.Test(t) - - t.Cleanup(func() { mock.AssertExpectations(t) }) - - return mock -} diff --git a/internal/rule/node.go b/internal/rule/node.go deleted file mode 100644 index c859819..0000000 --- a/internal/rule/node.go +++ /dev/null @@ -1,55 +0,0 @@ -package rule - -import "waffle/internal/request" - -type node interface { - Eval(r request.Wrapper) bool -} - -type and struct { - left, right node -} - -func (a and) Eval(r request.Wrapper) bool { - return a.left.Eval(r) && a.right.Eval(r) -} - -func (a and) SetChild(left, right node) { - a.left = left - a.right = right -} - -type or struct { - left, right node -} - -func (o or) Eval(r request.Wrapper) bool { - return o.left.Eval(r) || o.right.Eval(r) -} - -func (o or) SetChild(left, right node) { - o.left = left - o.right = right -} - -type eq struct { - left, right node -} - -func (e eq) Eval(r request.Wrapper) bool { - return e.left.Eval(r) == e.right.Eval(r) -} - -type gt struct { - valueFunc func(r request.Wrapper) (int, error) - check int -} - -func (g gt) Eval(r request.Wrapper) bool { - value, err := g.valueFunc(r) - if err != nil { - return false - } - - return value > g.check -} diff --git a/internal/rule/node_test.go b/internal/rule/node_test.go deleted file mode 100644 index f1c2320..0000000 --- a/internal/rule/node_test.go +++ /dev/null @@ -1,328 +0,0 @@ -package rule - -import ( - "testing" - - "github.com/stretchr/testify/mock" - - "waffle/internal/request" -) - -func Test_and_Eval(t *testing.T) { - type fields struct { - leftFunc func() node - rightFunc func() node - } - type args struct { - r request.Wrapper - } - tests := []struct { - name string - fields fields - args args - want bool - }{ - { - name: "both nodes true, true returned", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - }, - want: true, - }, - { - name: "one of nodes false, false returned", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - }, - want: false, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var left, right node - if tt.fields.leftFunc != nil { - left = tt.fields.leftFunc() - } - if tt.fields.rightFunc != nil { - right = tt.fields.rightFunc() - } - - a := and{ - left: left, - right: right, - } - if got := a.Eval(tt.args.r); got != tt.want { - t.Errorf("Eval() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_or_Eval(t *testing.T) { - type fields struct { - leftFunc func() node - rightFunc func() node - } - type args struct { - r request.Wrapper - } - tests := []struct { - name string - fields fields - args args - want bool - }{ - { - name: "first node true, returns true", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - }, - want: true, - }, - { - name: "both nodes false, returns false", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - }, - want: false, - }, - { - name: "first node false, second true, returns true", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - }, - want: true, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var left, right node - if tt.fields.leftFunc != nil { - left = tt.fields.leftFunc() - } - if tt.fields.rightFunc != nil { - right = tt.fields.rightFunc() - } - - o := or{ - left: left, - right: right, - } - if got := o.Eval(tt.args.r); got != tt.want { - t.Errorf("Eval() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_eq_Eval(t *testing.T) { - type fields struct { - leftFunc func() node - rightFunc func() node - } - type args struct { - r request.Wrapper - } - tests := []struct { - name string - fields fields - args args - want bool - }{ - { - name: "both nodes true, true returned", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - }, - want: true, - }, - { - name: "both nodes false, returns true", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - }, - want: true, - }, - { - name: "different nodes, false returned", - fields: fields{ - leftFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(true) - - return n - }, - rightFunc: func() node { - n := NewMocknode(t) - - n.EXPECT().Eval(mock.Anything).Return(false) - - return n - }, - }, - want: false, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var left, right node - if tt.fields.leftFunc != nil { - left = tt.fields.leftFunc() - } - if tt.fields.rightFunc != nil { - right = tt.fields.rightFunc() - } - - e := eq{ - left: left, - right: right, - } - if got := e.Eval(tt.args.r); got != tt.want { - t.Errorf("Eval() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_gt_Eval(t *testing.T) { - type fields struct { - valueFunc func(r request.Wrapper) (int, error) - check int - } - type args struct { - r request.Wrapper - } - tests := []struct { - name string - fields fields - args args - want bool - }{ - { - name: "value and check eq, false returned", - fields: fields{ - valueFunc: func(_ request.Wrapper) (int, error) { - return 69, nil - }, - check: 69, - }, - want: false, - }, - { - name: "value is greater than check, true returned", - fields: fields{ - valueFunc: func(_ request.Wrapper) (int, error) { - return 69, nil - }, - check: 1, - }, - want: true, - }, - { - name: "value is less than check, false returned", - fields: fields{ - valueFunc: func(_ request.Wrapper) (int, error) { - return 1, nil - }, - check: 69, - }, - want: false, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - g := gt{ - valueFunc: tt.fields.valueFunc, - check: tt.fields.check, - } - if got := g.Eval(tt.args.r); got != tt.want { - t.Errorf("Eval() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/internal/rule/predicate.go b/internal/rule/predicate.go deleted file mode 100644 index fdf1389..0000000 --- a/internal/rule/predicate.go +++ /dev/null @@ -1,44 +0,0 @@ -package rule - -import ( - "fmt" - - "waffle/internal/request" -) - -type ExpressionTreeBuilder interface { - BuildExpressionTree(variable, expression string) (expressionTree, error) -} - -type Builder interface { - Build(name, variable, expression string) (*Predicate, error) -} - -type Predicate struct { - Name string - Eval func(r request.Wrapper) bool -} - -type predicateBuilder struct { - treeBuilder ExpressionTreeBuilder -} - -func newPredicateBuilder() *predicateBuilder { - return &predicateBuilder{ - treeBuilder: newExpressionTreeBuilder(), - } -} - -var _ Builder = (*predicateBuilder)(nil) - -func (p *predicateBuilder) Build(name, variable, expression string) (*Predicate, error) { - tree, err := p.treeBuilder.BuildExpressionTree(variable, expression) - if err != nil { - return nil, fmt.Errorf("build expression tree: %w", err) - } - - return &Predicate{ - Name: name, - Eval: tree.Eval, - }, nil -} diff --git a/internal/rule/predicate_test.go b/internal/rule/predicate_test.go deleted file mode 100644 index e9e0af3..0000000 --- a/internal/rule/predicate_test.go +++ /dev/null @@ -1,81 +0,0 @@ -package rule - -import ( - "errors" - "github.com/stretchr/testify/mock" - "testing" -) - -func Test_predicateBuilder_Build(t *testing.T) { - andNode := &and{} - - type fields struct { - treeBuilderFunc func() ExpressionTreeBuilder - } - type args struct { - name string - variable string - expression string - } - tests := []struct { - name string - fields fields - args args - wantPredicate bool - wantErr bool - }{ - { - name: "builder returns error, error returned", - fields: fields{ - treeBuilderFunc: func() ExpressionTreeBuilder { - etb := NewMockExpressionTreeBuilder(t) - - etb.EXPECT().BuildExpressionTree(mock.Anything, mock.Anything).Return(nil, errors.New("error")) - - return etb - }, - }, - wantErr: true, - }, - { - name: "builder returns tree node, predicate returned", - fields: fields{ - treeBuilderFunc: func() ExpressionTreeBuilder { - etb := NewMockExpressionTreeBuilder(t) - - etb.EXPECT(). - BuildExpressionTree(mock.Anything, mock.Anything). - Return(andNode, nil) - - return etb - }, - }, - args: args{ - name: "test is test", - variable: "test", - expression: "test => LEN(test.payload) > 0", - }, - wantPredicate: true, - }, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - var treeBuilder ExpressionTreeBuilder - if tt.fields.treeBuilderFunc != nil { - treeBuilder = tt.fields.treeBuilderFunc() - } - - p := &predicateBuilder{ - treeBuilder: treeBuilder, - } - got, err := p.Build(tt.args.name, tt.args.variable, tt.args.expression) - if (err != nil) != tt.wantErr { - t.Errorf("Build() error = %v, wantErr %v", err, tt.wantErr) - return - } - if (got != nil) != tt.wantPredicate { - t.Errorf("Build() got = %v, want %v", got, tt.wantPredicate) - } - }) - } -} diff --git a/internal/rule/tokenizer.go b/internal/rule/tokenizer.go deleted file mode 100644 index f64fb4e..0000000 --- a/internal/rule/tokenizer.go +++ /dev/null @@ -1,201 +0,0 @@ -package rule - -import ( - "fmt" - "unicode" -) - -var ( - tokenVariable = "var" - tokenFunction = "func" - tokenNumber = "token_number" - tokenField = "field" - tokenSpace = "space" - - tokenLParen = "(" - tokenRParen = ")" - tokenDot = "." - tokenDoubleAmpersand = "&&" - tokenMoreThan = ">" - tokenLessThan = "<" - tokenSingleApostrophe = "'" - tokenOr = "||" - tokenEqual = "==" - tokenNotEqual = "!=" - tokensOperators = []string{ - tokenDot, - tokenDoubleAmpersand, - tokenMoreThan, - tokenLessThan, - tokenSingleApostrophe, - tokenOr, - tokenEqual, - tokenNotEqual, - } - - methodLen = "LEN" - methodFormat = "FORMAT" - methods = []string{ - methodLen, - methodFormat, - } - - fieldPayload = "payload" - fieldHeaders = "headers" - fields = []string{ - fieldPayload, - fieldHeaders, - } -) - -type Token struct { - Name string - Value string -} - -type tokenizer struct { -} - -var _ Tokenizer = (*tokenizer)(nil) - -func (t *tokenizer) BuildTokens(variable string, expression string) ([]Token, error) { - var ( - tokens []Token - match []rune - ) - - runeExpression := []rune(expression) - - for i, r := range runeExpression { - match = append(match, r) - - v, ok := getFunction(match) - if ok { - tokens = append(tokens, Token{ - Name: tokenFunction, - Value: v, - }) - - match = []rune{} - continue - } - - if isVariable(match, variable) && len(expression) >= i+1 && runeExpression[i+1] == '.' { - tokens = append(tokens, Token{ - Name: tokenVariable, - Value: string(match), - }) - - match = []rune{} - continue - } - - if v, ok := getSpecialCharacter(match); ok { - tokens = append(tokens, Token{ - Name: v, - Value: v, - }) - - match = []rune{} - continue - } - - if v, ok := getField(match); ok { - tokens = append(tokens, Token{ - Name: tokenField, - Value: v, - }) - - match = []rune{} - continue - } - - if unicode.IsSpace(match[0]) { - tokens = append(tokens, Token{ - Name: tokenSpace, - Value: tokenSpace, - }) - - match = []rune{} - continue - } - - if unicode.IsNumber(match[0]) { - tokens = append(tokens, Token{ - Name: tokenNumber, - Value: string(match[0]), - }) - - match = []rune{} - continue - } - - if string(match[0]) == tokenLParen { - tokens = append(tokens, Token{ - Name: tokenLParen, - Value: tokenLParen, - }) - - match = []rune{} - continue - } - - if string(match[0]) == tokenRParen { - tokens = append(tokens, Token{ - Name: tokenRParen, - Value: tokenRParen, - }) - - match = []rune{} - continue - } - } - - if len(match) > 0 { - return nil, fmt.Errorf("there are still unmatched characters") - } - - return tokens, nil -} - -func getFunction(match []rune) (string, bool) { - strMatch := string(match) - - for _, method := range methods { - if method == strMatch { - return method, true - } - } - - return "", false -} - -func isVariable(match []rune, variable string) bool { - strMatch := string(match) - - return strMatch == variable -} - -func getSpecialCharacter(match []rune) (string, bool) { - strMatch := string(match) - - for _, sch := range tokensOperators { - if sch == strMatch { - return sch, true - } - } - - return "", false -} - -func getField(match []rune) (string, bool) { - strMatch := string(match) - - for _, field := range fields { - if field == strMatch { - return field, true - } - } - - return "", false -} diff --git a/internal/rule/tokenizer_test.go b/internal/rule/tokenizer_test.go deleted file mode 100644 index cf823df..0000000 --- a/internal/rule/tokenizer_test.go +++ /dev/null @@ -1,144 +0,0 @@ -package rule - -import ( - "reflect" - "testing" -) - -var testTokens = []Token{ - { - Name: tokenFunction, - Value: "LEN", - }, - { - Name: tokenLParen, - Value: tokenLParen, - }, - { - Name: tokenVariable, - Value: "p", - }, - { - Name: tokenDot, - Value: tokenDot, - }, - { - Name: tokenField, - Value: "payload", - }, - { - Name: tokenRParen, - Value: tokenRParen, - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenMoreThan, - Value: ">", - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenNumber, - Value: "0", - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenDoubleAmpersand, - Value: "&&", - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenFunction, - Value: "LEN", - }, - { - Name: tokenLParen, - Value: tokenLParen, - }, - { - Name: tokenVariable, - Value: "p", - }, - { - Name: tokenDot, - Value: tokenDot, - }, - { - Name: tokenField, - Value: "headers", - }, - { - Name: tokenRParen, - Value: tokenRParen, - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenMoreThan, - Value: ">", - }, - { - Name: tokenSpace, - Value: tokenSpace, - }, - { - Name: tokenNumber, - Value: "0", - }, -} - -func Test_tokenizer_BuildTokens(t1 *testing.T) { - type args struct { - variable string - expression string - } - tests := []struct { - name string - args args - want []Token - wantErr bool - }{ - { - name: "wrong predicate, error returned", - args: args{ - variable: "add", - expression: "LANER(add.format) == 'json'", - }, - wantErr: true, - }, - { - name: "slice of tokens returned", - args: args{ - variable: "p", - expression: "LEN(p.payload) > 0 && LEN(p.headers) > 0", - }, - want: testTokens, - }, - } - for _, tt := range tests { - t1.Run(tt.name, func(t1 *testing.T) { - t := &tokenizer{} - got, err := t.BuildTokens(tt.args.variable, tt.args.expression) - if (err != nil) != tt.wantErr { - t1.Errorf("BuildTokens() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t1.Errorf("BuildTokens() got = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/pkg/permission/windows.go b/pkg/permission/windows.go deleted file mode 100644 index 00cf5cf..0000000 --- a/pkg/permission/windows.go +++ /dev/null @@ -1,30 +0,0 @@ -package permission - -//import ( -// "fmt" -// "golang.org/x/sys/windows" -// "os" -// "strings" -// "syscall" -//) -// -//func RunMeElevated() error { -// verb := "runas" -// exe, _ := os.Executable() -// cwd, _ := os.Getwd() -// args := strings.Join(os.Args[1:], " ") -// -// verbPtr, _ := syscall.UTF16PtrFromString(verb) -// exePtr, _ := syscall.UTF16PtrFromString(exe) -// cwdPtr, _ := syscall.UTF16PtrFromString(cwd) -// argPtr, _ := syscall.UTF16PtrFromString(args) -// -// var showCmd int32 = 1 //SW_NORMAL -// -// err := windows.ShellExecute(0, verbPtr, exePtr, argPtr, cwdPtr, showCmd) -// if err != nil { -// return fmt.Errorf("windows shell execute: %w", err) -// } -// -// return nil -//} From 2f1d9ac8b13b3520f246846c169d9a57c43f2575 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 11:56:33 +0200 Subject: [PATCH 23/27] Repaired taskfile and added description to all tasks --- Taskfile.yaml | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/Taskfile.yaml b/Taskfile.yaml index 9a97fb0..95ec2d6 100644 --- a/Taskfile.yaml +++ b/Taskfile.yaml @@ -1,10 +1,11 @@ version: '3' env: - IMAGE_NAME: '{{ .IMAGE_NAME | default "/:>" }}' + IMAGE_NAME: '{{ .IMAGE_NAME | default "/:" }}' tasks: certs_windows: + desc: "Generate SSL certificates for Windows environment using OpenSSL, including CA and server certificates." cmds: - mkdir -p "./cmd/proxy/.cert" - openssl ecparam -out ./cmd/proxy/.cert/ca.key -name prime256v1 -genkey @@ -16,6 +17,7 @@ tasks: - openssl x509 -in ./cmd/proxy/.cert/server.crt -text -noout certs: + desc: "Generate SSL certificates for non-Windows environments using OpenSSL, including CA and server certificates." cmds: - mkdir -p "./cmd/proxy/.cert" - openssl ecparam -out ./cmd/proxy/.cert/ca.key -name prime256v1 -genkey @@ -27,13 +29,16 @@ tasks: - openssl x509 -in ./cmd/proxy/.cert/server.crt -text -noout mocks: + desc: "Generate mock implementations for testing using Mockery." cmds: - mockery dockerBuild: + desc: "Build a Docker image using the specified Dockerfile and tag it with the provided image name." cmds: - - docker build -t {{ .IMAGE_NAME }] -f .\build\Dockerfile . + - docker build -t {{ .IMAGE_NAME }} -f .\build\Dockerfile . dockerPush: + desc: "Push the Docker image to the remote registry with the specified image name." cmds: - - docker push {{ .IMAGE_NAME }] + - docker push {{ .IMAGE_NAME }} From 9751d027072d5fb99e4ecbc418f60dea4c0dbca8 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 11:58:59 +0200 Subject: [PATCH 24/27] fix: Building application binary instead of docker, since we should login in registry then --- .github/workflows/go.yml | 4 ++-- Taskfile.yaml | 5 +++++ 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index f32a8aa..d96628b 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -33,5 +33,5 @@ jobs: - name: Install go-task to build docker image run: sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -d -b ~/.local/bin - - name: Build Waffle docker image - run: task dockerBuild \ No newline at end of file + - name: Build Waffle using Golang + run: task docker \ No newline at end of file diff --git a/Taskfile.yaml b/Taskfile.yaml index 95ec2d6..79d835d 100644 --- a/Taskfile.yaml +++ b/Taskfile.yaml @@ -33,6 +33,11 @@ tasks: cmds: - mockery + build: + desc: "Build a project using Golang" + cmds: + - go build -o /out/app ./cmd/proxy + dockerBuild: desc: "Build a Docker image using the specified Dockerfile and tag it with the provided image name." cmds: From 042fa0e11ebd6811cec7a5b1b2f6cf64fc5ef665 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 11:59:20 +0200 Subject: [PATCH 25/27] fixed task name from dockerBuild to build --- .github/workflows/go.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index d96628b..db01701 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -34,4 +34,4 @@ jobs: run: sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -d -b ~/.local/bin - name: Build Waffle using Golang - run: task docker \ No newline at end of file + run: task build \ No newline at end of file From a0725ea2fdf813f087ab397663e96f72049a2fc6 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 12:00:42 +0200 Subject: [PATCH 26/27] fix: Changed output path in build task --- Taskfile.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Taskfile.yaml b/Taskfile.yaml index 79d835d..6bfe8bf 100644 --- a/Taskfile.yaml +++ b/Taskfile.yaml @@ -36,7 +36,7 @@ tasks: build: desc: "Build a project using Golang" cmds: - - go build -o /out/app ./cmd/proxy + - go build -o ./out/app ./cmd/proxy dockerBuild: desc: "Build a Docker image using the specified Dockerfile and tag it with the provided image name." From d1eaa5fd03dd60e5cada6f02c40e48999f443111 Mon Sep 17 00:00:00 2001 From: Oleksand Matviienko Date: Thu, 24 Oct 2024 12:06:08 +0200 Subject: [PATCH 27/27] Changed flag name to avoid duplication and added better logging. --- cmd/proxy/main.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/cmd/proxy/main.go b/cmd/proxy/main.go index d0d3e34..8009bc3 100644 --- a/cmd/proxy/main.go +++ b/cmd/proxy/main.go @@ -24,7 +24,7 @@ func main() { visualizeServerPort string proxyServerPort string ) - flag.StringVar(&visualizeServerPort, "p", "8081", "Port for server to listen on") + flag.StringVar(&visualizeServerPort, "vp", "8080", "Port for visualizer server to listen on") flag.StringVar(&proxyServerPort, "p", "8081", "Port for server to listen on") quit := make(chan os.Signal) @@ -37,7 +37,8 @@ func main() { } }() - log.Println("Server started on :8080") + log.Println("Visualizer server started on ", visualizeServerPort) + log.Println("Proxy server started on ", proxyServerPort) <-quit log.Println("Shutdown signal received, shutting down gracefully...")