diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index bd1e811..70243ef 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -31,5 +31,5 @@ jobs: go get gopkg.in/check.v1 - name: Build run: go build -v . - # - name: Test - # run: go test -v ./... + - name: Test + run: go test -v ./... diff --git a/.gitignore b/.gitignore index d89d452..88efece 100644 --- a/.gitignore +++ b/.gitignore @@ -7,4 +7,5 @@ docs/kubernetes/oingress.yaml docs/kubernetes/ns.yaml docs/kubernetes/ns-config.yaml docs/kubernetes/kubeconfig.yaml +docs/kubernetes/test.sh .DS_Store diff --git a/Dockerfile b/Dockerfile index 2f009a5..02ebfd3 100644 --- a/Dockerfile +++ b/Dockerfile @@ -9,7 +9,6 @@ FROM alpine:latest RUN apk --no-cache add ca-certificates WORKDIR /root/ COPY --from=build /go/src/github.com/natron-io/tenant-api/tenant-api ./ -COPY --from=build /go/src/github.com/natron-io/tenant-api/views ./views -COPY --from=build /go/src/github.com/natron-io/tenant-api/static ./static +COPY --from=build /go/src/github.com/natron-io/tenant-api/web ./web EXPOSE 8000 CMD ["./tenant-api"] \ No newline at end of file diff --git a/README.md b/README.md index 935ecf6..38417fa 100644 --- a/README.md +++ b/README.md @@ -21,7 +21,7 @@ You can also sync your slack broadcast channel to present some important informa ## api -#### `GET` +### `GET` > **important:** for authenticated access you need to provide the `Authorization` header with the `Bearer` token. You can add `` in front of the path to get the tenant specific data (of everything). @@ -47,7 +47,8 @@ You can add `` in front of the path to get the tenant specific data (of `/api/v1//costs/cpu` - Get the CPU costs by CPU \ `/api/v1//costs/memory` - Get the memory costs by Memory \ `/api/v1//costs/storage` - Get the storage costs by StorageClass \ -`/api/v1//costs/ingress` - Get the ingress costs by tenant +`/api/v1//costs/ingress` - Get the ingress costs by tenant \ +`/api/v1//costs/currentmonth` - Get a list of the average cost consumption for the current month ##### tenant resource quotas `/api/v1//quotas/cpu` - Get the CPU resource Quota by the label defined via env \ @@ -55,7 +56,7 @@ You can add `` in front of the path to get the tenant specific data (of `/api/v1//quotas/storage` - Get the storage resource Quota for each storage class by the label**s** defined via env -#### `POST` +### `POST` ##### auth You can send the github code with json body `{"github_code": "..."}` to the `/login/github` endpoint. @@ -65,6 +66,7 @@ You can send the github code with json body `{"github_code": "..."}` to the `/lo ### general `CORS` - CORS middleware for Fiber that that can be used to enable Cross-Origin Resource Sharing with various options. (e.g. "https://example.com, https://example2.com") +`DEBUG` - Disables API authentication / authorization (tenants doesn't work anymore, bc jwt won't get validated and parsed) ### GitHub > There are two ways for authenticating with GitHub. You can authenticate without a dashboard, so the github callback url is not the same as the dashboard. @@ -94,6 +96,15 @@ You can send the github code with json body `{"github_code": "..."}` to the `/lo `INGRESS_COST_PER_DOMAIN` - Calculates only ingress per domain.tld format *optional* (default: false) \ `EXCLUDE_INGRESS_VCLUSTER` - Excludes the vcluster ingress resource to expose the vcluster Kubernetes API. Name of the ingress must contain the string "vcluster" *optional* (default: false) +#### cost persistency +`COST_PERSISTENCY` - if set to true, database persistency configuration must be set *optional* (default: false) \ +`COST_PERSISTENCY_INTERVAL` - interval in seconds to log cost data into database *optional* (default: 3600) \ +`DB_HOST` - postgresql db host *optional* (default: localhost) \ +`DB_PORT` - postgresql db port *optional* (default: 5432) \ +`DB_USER` - postgresql db user *optional* (default: postgres) \ +`DB_PASSWORD` - postgresql db password *optional* (default: postgres) \ +`DB_NAME` - postgresql db name *optional* (default: postgres) \ +`DB_SSLMODE` - postgresql sslmode *optional* (default: postgres) ### resource quotas It will get the resource quotas defined in the tenant namespace with the exact name of the tenant. diff --git a/controllers/authController.go b/controllers/authController.go index b1297ca..691a5fc 100644 --- a/controllers/authController.go +++ b/controllers/authController.go @@ -116,6 +116,10 @@ func CheckAuth(c *fiber.Ctx) []string { var token *jwt.Token var tokenString string + if util.DEBUG { + return []string{"debug"} + } + // get bearer token from header bearerToken := c.Get("Authorization") diff --git a/controllers/costController.go b/controllers/costController.go index af8b68c..dd55839 100644 --- a/controllers/costController.go +++ b/controllers/costController.go @@ -2,7 +2,10 @@ package controllers import ( "github.com/gofiber/fiber/v2" + "github.com/natron-io/tenant-api/database" + "github.com/natron-io/tenant-api/models" "github.com/natron-io/tenant-api/util" + "gorm.io/gorm" ) // GetCPUCostSum returns the cpu cost sum per tenant @@ -196,3 +199,90 @@ func GetIngressCostSum(c *fiber.Ctx) error { return c.JSON(tenantsIngressCostsPerDomain[tenant]) } } + +func GetMonthlyCostSum(c *fiber.Ctx) error { + + util.InfoLogger.Printf("%s %s %s", c.IP(), c.Method(), c.Path()) + tenant := c.Params("tenant") + tenants := CheckAuth(c) + if len(tenants) == 0 { + return c.Status(401).JSON(fiber.Map{ + "message": "Unauthorized", + }) + } + if tenant != "" && !util.Contains(tenant, tenants) { + return c.Status(403).JSON(fiber.Map{ + "message": "Forbidden", + }) + } + + db := database.DBConn + + // search tenants in database GitHubTeamSlug + dbTenant := models.Tenant{ + GitHubTeamSlug: tenant, + } + + err := db.Where(&dbTenant).First(&dbTenant).Error + if err != nil { + if err == gorm.ErrRecordNotFound { + return c.Status(404).JSON(fiber.Map{ + "message": "Not Found", + }) + } + util.ErrorLogger.Printf("%s", err) + return c.Status(500).JSON(fiber.Map{ + "message": "Internal Server Error", + }) + } + + // get cpuCosts by month and tenant + cpuCosts, err := util.GetCPUCostsByMonthAndTenant(dbTenant) + if err != nil { + return c.Status(500).JSON(fiber.Map{ + "message": "Internal Server Error", + }) + } + + // get memoryCosts by month and tenant + memoryCosts, err := util.GetMemoryCostsByMonthAndTenant(dbTenant) + if err != nil { + return c.Status(500).JSON(fiber.Map{ + "message": "Internal Server Error", + }) + } + + // get storageCosts by month and tenant + storageCosts, err := util.GetStorageCostsByMonthAndTenant(dbTenant) + if err != nil { + return c.Status(500).JSON(fiber.Map{ + "message": "Internal Server Error", + }) + } + + var storageClassCostSum map[string]float64 + for _, storageClass := range storageCosts { + if storageClassCostSum == nil { + storageClassCostSum = make(map[string]float64) + } + storageClassCostSum[storageClass.StorageClass] = storageClassCostSum[storageClass.StorageClass] + storageClass.Value + } + + ingressCosts, err := util.GetIngressCostsByMonthAndTenant(dbTenant) + if err != nil { + return c.Status(500).JSON(fiber.Map{ + "message": "Internal Server Error", + }) + } + + // create JSON object for each cost + costsJSON := make(map[string]float64) + costsJSON["cpu"] = cpuCosts.Value + costsJSON["memory"] = memoryCosts.Value + for storageClass, value := range storageClassCostSum { + costsJSON[storageClass] = value + } + costsJSON["ingress"] = ingressCosts.Value + + return c.JSON(costsJSON) +} diff --git a/controllers/doc.go b/controllers/doc.go index cc9b237..f58ef47 100644 --- a/controllers/doc.go +++ b/controllers/doc.go @@ -1,5 +1,5 @@ /* -Copyright 2022 Jan Lauber +Copyright 2022 Netrics AG Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. diff --git a/database/database.go b/database/database.go new file mode 100644 index 0000000..fcce6f9 --- /dev/null +++ b/database/database.go @@ -0,0 +1,44 @@ +package database + +import ( + "fmt" + + "github.com/natron-io/tenant-api/models" + "gorm.io/driver/postgres" + "gorm.io/gorm" +) + +var ( + DBConn *gorm.DB + err error + DB_HOST string + DB_PORT string + DB_USER string + DB_PASSWORD string + DB_NAME string + DB_SSLMODE string +) + +func InitDB() error { + // Connect to the database + dbUri := fmt.Sprintf("host=%s user=%s password=%s dbname=%s port=%s sslmode=%s", DB_HOST, DB_USER, DB_PASSWORD, DB_NAME, DB_PORT, DB_SSLMODE) + DBConn, err = gorm.Open(postgres.Open(dbUri), &gorm.Config{}) + if err != nil { + return err + } + + // Migrate the schema + err = DBConn.AutoMigrate( + &models.Tenant{}, + &models.CPUCost{}, + &models.MemoryCost{}, + &models.StorageCost{}, + &models.IngressCost{}, + &models.MonthlyCost{}, + ) + if err != nil { + return err + } + + return nil +} diff --git a/database/doc.go b/database/doc.go new file mode 100644 index 0000000..b2287a6 --- /dev/null +++ b/database/doc.go @@ -0,0 +1,18 @@ +/* +Copyright 2022 Netrics AG + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +// Package controllers is responsible to set up the controllers for the http handlers +package database diff --git a/docs/kubernetes/ingress.yaml b/docs/kubernetes/ingress.yaml index 76ffd3c..4090b08 100644 --- a/docs/kubernetes/ingress.yaml +++ b/docs/kubernetes/ingress.yaml @@ -18,7 +18,7 @@ spec: name: tenant-api port: number: 8000 - - host: api2.example.com + - host: api2.example4.com http: paths: - path: / @@ -68,4 +68,4 @@ spec: service: name: tenant-api port: - number: 8000 \ No newline at end of file + number: 8000 diff --git a/docs/kubernetes/postgresql-values.yaml b/docs/kubernetes/postgresql-values.yaml new file mode 100644 index 0000000..d3a9e8c --- /dev/null +++ b/docs/kubernetes/postgresql-values.yaml @@ -0,0 +1,5 @@ +# Test Helm Values +global: + postgresql: + auth: + postgresPassword: "12345678" \ No newline at end of file diff --git a/go.mod b/go.mod index eaa668f..9d9d2e9 100644 --- a/go.mod +++ b/go.mod @@ -2,7 +2,11 @@ module github.com/natron-io/tenant-api go 1.17 -require k8s.io/apimachinery v0.23.1 +require ( + github.com/slack-go/slack v0.10.1 + k8s.io/api v0.23.1 + k8s.io/apimachinery v0.23.1 +) require ( github.com/andybalholm/brotli v1.0.2 // indirect @@ -10,12 +14,22 @@ require ( github.com/golang/protobuf v1.5.2 // indirect github.com/googleapis/gnostic v0.5.5 // indirect github.com/gorilla/websocket v1.4.2 // indirect + github.com/jackc/chunkreader/v2 v2.0.1 // indirect + github.com/jackc/pgconn v1.10.1 // indirect + github.com/jackc/pgio v1.0.0 // indirect + github.com/jackc/pgpassfile v1.0.0 // indirect + github.com/jackc/pgproto3/v2 v2.2.0 // indirect + github.com/jackc/pgservicefile v0.0.0-20200714003250-2b9c44734f2b // indirect + github.com/jackc/pgtype v1.9.1 // indirect + github.com/jackc/pgx/v4 v4.14.1 // indirect + github.com/jinzhu/inflection v1.0.0 // indirect + github.com/jinzhu/now v1.1.4 // indirect github.com/klauspost/compress v1.13.4 // indirect github.com/pkg/errors v0.9.1 // indirect - github.com/slack-go/slack v0.10.1 // indirect github.com/valyala/bytebufferpool v1.0.0 // indirect github.com/valyala/fasthttp v1.31.0 // indirect github.com/valyala/tcplisten v1.0.0 // indirect + golang.org/x/crypto v0.0.0-20210921155107-089bfa567519 // indirect golang.org/x/oauth2 v0.0.0-20210819190943-2bc19b11175f // indirect golang.org/x/sys v0.0.0-20210831042530-f4d43177bf5e // indirect golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b // indirect @@ -23,7 +37,6 @@ require ( google.golang.org/appengine v1.6.7 // indirect google.golang.org/protobuf v1.27.1 // indirect gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect - k8s.io/api v0.23.1 // indirect k8s.io/kube-openapi v0.0.0-20211115234752-e816edb12b65 // indirect sigs.k8s.io/yaml v1.2.0 // indirect ) @@ -43,6 +56,8 @@ require ( golang.org/x/text v0.3.7 // indirect gopkg.in/inf.v0 v0.9.1 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect + gorm.io/driver/postgres v1.3.1 + gorm.io/gorm v1.23.2 k8s.io/client-go v0.23.1 k8s.io/klog/v2 v2.30.0 // indirect k8s.io/utils v0.0.0-20210930125809-cb0fa318a74b // indirect diff --git a/go.sum b/go.sum index 31d3e42..2284a3c 100644 --- a/go.sum +++ b/go.sum @@ -50,6 +50,8 @@ github.com/CloudyKit/fastprinter v0.0.0-20200109182630-33d98a066a53/go.mod h1:+3 github.com/CloudyKit/jet/v6 v6.1.0/go.mod h1:d3ypHeIRNo2+XyqnGA8s+aphtcVpjP5hPwP/Lzo7Ro4= github.com/Joker/hpp v1.0.0/go.mod h1:8x5n+M1Hp5hC0g8okX3sR3vFQwynaX/UgSOM9MeBKzY= github.com/Joker/jade v1.1.1/go.mod h1:C5O3w7HbsWdb9ik1puKS81QsllcBd+CXRVCbXFwSdsE= +github.com/Masterminds/semver/v3 v3.1.1 h1:hLg3sBzpNErnxhQtUy/mmLR2I9foDujNK030IGemrRc= +github.com/Masterminds/semver/v3 v3.1.1/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs= github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ= github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= @@ -72,9 +74,14 @@ github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDk github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= +github.com/cockroachdb/apd v1.1.0 h1:3LFP3629v+1aKXU5Q37mxmRxX/pIu1nijXydLShEq5I= +github.com/cockroachdb/apd v1.1.0/go.mod h1:8Sl8LxpKi29FqWXR16WEFZRNSz3SoPzUzeMeY4+DwBQ= github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= +github.com/coreos/go-systemd v0.0.0-20190719114852-fd7a80b32e1f/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= +github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= @@ -103,6 +110,8 @@ github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeME 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/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= +github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas= github.com/go-logr/logr v0.2.0/go.mod h1:z6/tIYblkpsD+a4lm/fGIIU9mZ+XfAiaFtq7xTgseGU= github.com/go-logr/logr v1.2.0 h1:QK40JKJyMdUDz+h+xvCsru/bJhvG0UxvePV0ufL/AcE= @@ -111,12 +120,16 @@ github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34 github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= github.com/go-openapi/jsonreference v0.19.3/go.mod h1:rjx6GuL8TTa9VaixXglHmQmIL98+wF9xc8zWvFonSJ8= github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/go-test/deep v1.0.4 h1:u2CU3YKy9I2pmu9pX0eq50wCgjfGIt539SqR7FbHiho= github.com/go-test/deep v1.0.4/go.mod h1:wGDj63lr65AM2AQyKZd/NYHGb0R+1RLqB8NKt3aSFNA= github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/gofiber/fiber/v2 v2.24.0 h1:18rpLoQMJBVlLtX/PwgHj3hIxPSeWfN1YeDJ2lEnzjU= github.com/gofiber/fiber/v2 v2.24.0/go.mod h1:MR1usVH3JHYRyQwMe2eZXRSZHRX38fkV+A7CPB+DlDQ= github.com/gofiber/template v1.6.21 h1:oIqrnpHnCWhbdBIme5AHBsM7c93H2vrb6Xyloj7UZoU= github.com/gofiber/template v1.6.21/go.mod h1:1ORcy7sg3WW/YVeAyeGkcCtT63HjEj3+qxHNz+oZoFk= +github.com/gofrs/uuid v4.0.0+incompatible h1:1SD/1F5pU8p29ybwgQSwpQk+mwdRrXCYuPhW6m+TnJw= +github.com/gofrs/uuid v4.0.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM= github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY= @@ -223,6 +236,59 @@ github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1: github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA= github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= +github.com/jackc/chunkreader v1.0.0 h1:4s39bBR8ByfqH+DKm8rQA3E1LHZWB9XWcrz8fqaZbe0= +github.com/jackc/chunkreader v1.0.0/go.mod h1:RT6O25fNZIuasFJRyZ4R/Y2BbhasbmZXF9QQ7T3kePo= +github.com/jackc/chunkreader/v2 v2.0.0/go.mod h1:odVSm741yZoC3dpHEUXIqA9tQRhFrgOHwnPIn9lDKlk= +github.com/jackc/chunkreader/v2 v2.0.1 h1:i+RDz65UE+mmpjTfyz0MoVTnzeYxroil2G82ki7MGG8= +github.com/jackc/chunkreader/v2 v2.0.1/go.mod h1:odVSm741yZoC3dpHEUXIqA9tQRhFrgOHwnPIn9lDKlk= +github.com/jackc/pgconn v0.0.0-20190420214824-7e0022ef6ba3/go.mod h1:jkELnwuX+w9qN5YIfX0fl88Ehu4XC3keFuOJJk9pcnA= +github.com/jackc/pgconn v0.0.0-20190824142844-760dd75542eb/go.mod h1:lLjNuW/+OfW9/pnVKPazfWOgNfH2aPem8YQ7ilXGvJE= +github.com/jackc/pgconn v0.0.0-20190831204454-2fabfa3c18b7/go.mod h1:ZJKsE/KZfsUgOEh9hBm+xYTstcNHg7UPMVJqRfQxq4s= +github.com/jackc/pgconn v1.8.0/go.mod h1:1C2Pb36bGIP9QHGBYCjnyhqu7Rv3sGshaQUvmfGIB/o= +github.com/jackc/pgconn v1.9.0/go.mod h1:YctiPyvzfU11JFxoXokUOOKQXQmDMoJL9vJzHH8/2JY= +github.com/jackc/pgconn v1.9.1-0.20210724152538-d89c8390a530/go.mod h1:4z2w8XhRbP1hYxkpTuBjTS3ne3J48K83+u0zoyvg2pI= +github.com/jackc/pgconn v1.10.1 h1:DzdIHIjG1AxGwoEEqS+mGsURyjt4enSmqzACXvVzOT8= +github.com/jackc/pgconn v1.10.1/go.mod h1:4z2w8XhRbP1hYxkpTuBjTS3ne3J48K83+u0zoyvg2pI= +github.com/jackc/pgio v1.0.0 h1:g12B9UwVnzGhueNavwioyEEpAmqMe1E/BN9ES+8ovkE= +github.com/jackc/pgio v1.0.0/go.mod h1:oP+2QK2wFfUWgr+gxjoBH9KGBb31Eio69xUb0w5bYf8= +github.com/jackc/pgmock v0.0.0-20190831213851-13a1b77aafa2/go.mod h1:fGZlG77KXmcq05nJLRkk0+p82V8B8Dw8KN2/V9c/OAE= +github.com/jackc/pgmock v0.0.0-20201204152224-4fe30f7445fd/go.mod h1:hrBW0Enj2AZTNpt/7Y5rr2xe/9Mn757Wtb2xeBzPv2c= +github.com/jackc/pgmock v0.0.0-20210724152146-4ad1a8207f65 h1:DadwsjnMwFjfWc9y5Wi/+Zz7xoE5ALHsRQlOctkOiHc= +github.com/jackc/pgmock v0.0.0-20210724152146-4ad1a8207f65/go.mod h1:5R2h2EEX+qri8jOWMbJCtaPWkrrNc7OHwsp2TCqp7ak= +github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= +github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= +github.com/jackc/pgproto3 v1.1.0 h1:FYYE4yRw+AgI8wXIinMlNjBbp/UitDJwfj5LqqewP1A= +github.com/jackc/pgproto3 v1.1.0/go.mod h1:eR5FA3leWg7p9aeAqi37XOTgTIbkABlvcPB3E5rlc78= +github.com/jackc/pgproto3/v2 v2.0.0-alpha1.0.20190420180111-c116219b62db/go.mod h1:bhq50y+xrl9n5mRYyCBFKkpRVTLYJVWeCc+mEAI3yXA= +github.com/jackc/pgproto3/v2 v2.0.0-alpha1.0.20190609003834-432c2951c711/go.mod h1:uH0AWtUmuShn0bcesswc4aBTWGvw0cAxIJp+6OB//Wg= +github.com/jackc/pgproto3/v2 v2.0.0-rc3/go.mod h1:ryONWYqW6dqSg1Lw6vXNMXoBJhpzvWKnT95C46ckYeM= +github.com/jackc/pgproto3/v2 v2.0.0-rc3.0.20190831210041-4c03ce451f29/go.mod h1:ryONWYqW6dqSg1Lw6vXNMXoBJhpzvWKnT95C46ckYeM= +github.com/jackc/pgproto3/v2 v2.0.6/go.mod h1:WfJCnwN3HIg9Ish/j3sgWXnAfK8A9Y0bwXYU5xKaEdA= +github.com/jackc/pgproto3/v2 v2.1.1/go.mod h1:WfJCnwN3HIg9Ish/j3sgWXnAfK8A9Y0bwXYU5xKaEdA= +github.com/jackc/pgproto3/v2 v2.2.0 h1:r7JypeP2D3onoQTCxWdTpCtJ4D+qpKr0TxvoyMhZ5ns= +github.com/jackc/pgproto3/v2 v2.2.0/go.mod h1:WfJCnwN3HIg9Ish/j3sgWXnAfK8A9Y0bwXYU5xKaEdA= +github.com/jackc/pgservicefile v0.0.0-20200714003250-2b9c44734f2b h1:C8S2+VttkHFdOOCXJe+YGfa4vHYwlt4Zx+IVXQ97jYg= +github.com/jackc/pgservicefile v0.0.0-20200714003250-2b9c44734f2b/go.mod h1:vsD4gTJCa9TptPL8sPkXrLZ+hDuNrZCnj29CQpr4X1E= +github.com/jackc/pgtype v0.0.0-20190421001408-4ed0de4755e0/go.mod h1:hdSHsc1V01CGwFsrv11mJRHWJ6aifDLfdV3aVjFF0zg= +github.com/jackc/pgtype v0.0.0-20190824184912-ab885b375b90/go.mod h1:KcahbBH1nCMSo2DXpzsoWOAfFkdEtEJpPbVLq8eE+mc= +github.com/jackc/pgtype v0.0.0-20190828014616-a8802b16cc59/go.mod h1:MWlu30kVJrUS8lot6TQqcg7mtthZ9T0EoIBFiJcmcyw= +github.com/jackc/pgtype v1.8.1-0.20210724151600-32e20a603178/go.mod h1:C516IlIV9NKqfsMCXTdChteoXmwgUceqaLfjg2e3NlM= +github.com/jackc/pgtype v1.9.1 h1:MJc2s0MFS8C3ok1wQTdQxWuXQcB6+HwAm5x1CzW7mf0= +github.com/jackc/pgtype v1.9.1/go.mod h1:LUMuVrfsFfdKGLw+AFFVv6KtHOFMwRgDDzBt76IqCA4= +github.com/jackc/pgx/v4 v4.0.0-20190420224344-cc3461e65d96/go.mod h1:mdxmSJJuR08CZQyj1PVQBHy9XOp5p8/SHH6a0psbY9Y= +github.com/jackc/pgx/v4 v4.0.0-20190421002000-1b8f0016e912/go.mod h1:no/Y67Jkk/9WuGR0JG/JseM9irFbnEPbuWV2EELPNuM= +github.com/jackc/pgx/v4 v4.0.0-pre1.0.20190824185557-6972a5742186/go.mod h1:X+GQnOEnf1dqHGpw7JmHqHc1NxDoalibchSk9/RWuDc= +github.com/jackc/pgx/v4 v4.12.1-0.20210724153913-640aa07df17c/go.mod h1:1QD0+tgSXP7iUjYm9C1NxKhny7lq6ee99u/z+IHFcgs= +github.com/jackc/pgx/v4 v4.14.1 h1:71oo1KAGI6mXhLiTMn6iDFcp3e7+zon/capWjl2OEFU= +github.com/jackc/pgx/v4 v4.14.1/go.mod h1:RgDuE4Z34o7XE92RpLsvFiOEfrAUT0Xt2KxvX73W06M= +github.com/jackc/puddle v0.0.0-20190413234325-e4ced69a3a2b/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dvjSWk= +github.com/jackc/puddle v0.0.0-20190608224051-11cab39313c9/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dvjSWk= +github.com/jackc/puddle v1.1.3/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dvjSWk= +github.com/jackc/puddle v1.2.0/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dvjSWk= +github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= +github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/jinzhu/now v1.1.4 h1:tHnRBy1i5F2Dh8BAFxqFzxKqqvezXrL2OW1TnX+Mlas= +github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= @@ -234,18 +300,31 @@ github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/klauspost/compress v1.13.4 h1:0zhec2I8zGnjWcKyLl6i3gPqKANCCn5e9xmviEEeX6s= github.com/klauspost/compress v1.13.4/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= +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.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/pty v1.1.8/go.mod h1:O1sed60cT9XZ5uDucP5qwvh+TE3NnUj51EiZO/lmSfw= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/lib/pq v1.0.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= +github.com/lib/pq v1.1.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= +github.com/lib/pq v1.2.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= +github.com/lib/pq v1.10.2 h1:AqzbZs4ZoCBp+GtejcpCpcxM3zlSMx29dXbUSeVtJb8= +github.com/lib/pq v1.10.2/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o= github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= +github.com/mattn/go-colorable v0.1.1/go.mod h1:FuOcm+DKB9mbwrcAfNl7/TZVBZ6rcnceauSikq3lYCQ= +github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.5/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= +github.com/mattn/go-isatty v0.0.7/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= +github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= github.com/mattn/go-slim v0.0.0-20200618151855-bde33eecb5ee/go.mod h1:ma9TUJeni8LGZMJvOwbAv/FOwiwqIMQN570LnpqCBSM= github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= @@ -293,10 +372,19 @@ github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndr github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= 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.13.0/go.mod h1:YbFCdg8HfsridGWAh22vktObvhZbQsZXe4/zB0OKkWU= +github.com/rs/zerolog v1.15.0/go.mod h1:xYTKnLHcpfU2225ny5qZjxnj9NvkumZYjJHlAThCjNc= github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= +github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= +github.com/shopspring/decimal v0.0.0-20180709203117-cd690d0c9e24/go.mod h1:M+9NzErvs504Cn4c5DxATwIqPbtswREoFCre64PpcG4= +github.com/shopspring/decimal v1.2.0 h1:abSATXmQEYyShuxI4/vyW3tV1MrKAJzCZ/0zLUXYbsQ= +github.com/shopspring/decimal v1.2.0/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o= github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +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/slack-go/slack v0.10.1 h1:BGbxa0kMsGEvLOEoZmYs8T1wWfoZXwmQFBb6FgYCXUA= github.com/slack-go/slack v0.10.1/go.mod h1:wWL//kk0ho+FcQXcBTmEafUI5dz4qz5f4mMk8oIkioQ= github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= @@ -311,6 +399,8 @@ github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An github.com/spf13/viper v1.8.1/go.mod h1:o0Pch8wJ9BVSWGQMbra6iw0oQ5oktSIBaujf1rJH9Ns= github.com/stoewer/go-strcase v1.2.0/go.mod h1:IBiWB2sKIp3wVVQ3Y035++gc+knqhUQag1KpM8ahLw8= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= @@ -331,6 +421,7 @@ github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9de github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +github.com/zenazn/goji v0.9.0/go.mod h1:7S9M489iMyHBNxwZnk9/EHS098H4/F6TATF2mIxtB1Q= go.etcd.io/etcd/api/v3 v3.5.0/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= go.etcd.io/etcd/client/pkg/v3 v3.5.0/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= go.etcd.io/etcd/client/v2 v2.305.0/go.mod h1:h9puh54ZTgAKtEbut2oe9P4L/oqKCVB6xsXlzd7alYQ= @@ -341,19 +432,36 @@ go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= +go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= +go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= +go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= +go.uber.org/atomic v1.6.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= +go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= +go.uber.org/multierr v1.5.0/go.mod h1:FeouvMocqHpRaaGuG9EjoKcStLC43Zu/fmqdUMPcKYU= go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= +go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA= +go.uber.org/zap v1.9.1/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= +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.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo= 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-20190411191339-88737f569e3a/go.mod h1:WFFai1msRO1wXaEeE5yQxYXgSfI8pQAWXbQop6sCtWE= 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-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20201002170205-7f63de1d35b0/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20201203163018-be400aefbc4c/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= golang.org/x/crypto v0.0.0-20210513164829-c07d793c2f9a/go.mod h1:P+XmwS30IXTQdn5tA2iutPOUgjI07+tq3H3K9MVA1s8= +golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519 h1:7I4JAnoQBe7ZtJcBaYHi5UtiO8tQHbUSXxL+pnGRANg= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/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= @@ -405,6 +513,7 @@ golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190628185345-da137c7871d7/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-20190827160401-ba9fcec4b297/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= @@ -459,23 +568,30 @@ golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-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-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-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-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -507,6 +623,7 @@ golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210831042530-f4d43177bf5e h1:XMgFehsDnnLGtjvjOfqWSUzt0alpTR1RSEuznObga2c= golang.org/x/sys v0.0.0-20210831042530-f4d43177bf5e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b h1:9zKuko04nR4gjZ4+DNjHqRlAJqbJETHwiNKDqTfOjfE= golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= @@ -533,14 +650,18 @@ golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3 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-20190425163242-31fd60d6bfdc/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-20190823170909-c4a336ef6a2f/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-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-20191112195655-aa38f8e97acc/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= @@ -549,6 +670,7 @@ golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtn 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-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= @@ -579,6 +701,8 @@ golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4f golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/xerrors v0.0.0-20190410155217-1f06c39b4373/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20190513163551-3ee3066db522/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -698,6 +822,7 @@ gopkg.in/check.v1 v1.0.0-20200902074654-038fdea0a05b h1:QRR6H1YWRnHb4Y/HeNFCTJLF gopkg.in/check.v1 v1.0.0-20200902074654-038fdea0a05b/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= 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/inconshreveable/log15.v2 v2.0.0-20180818164646-67afb5ed74ec/go.mod h1:aPpfJ7XW+gOuirDoZ8gHhLh3kZ1B08FtV2bbmy7Jv3s= gopkg.in/inf.v0 v0.9.1 h1:73M5CoZyi3ZLMOyDlQh031Cx6N9NDJ2Vvfl76EDAgDc= gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= gopkg.in/ini.v1 v1.62.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= @@ -715,6 +840,11 @@ gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gorm.io/driver/postgres v1.3.1 h1:Pyv+gg1Gq1IgsLYytj/S2k7ebII3CzEdpqQkPOdH24g= +gorm.io/driver/postgres v1.3.1/go.mod h1:WwvWOuR9unCLpGWCL6Y3JOeBWvbKi6JLhayiVclSZZU= +gorm.io/gorm v1.23.1/go.mod h1:l2lP/RyAtc1ynaTjFksBde/O8v9oOGIApu2/xRitmZk= +gorm.io/gorm v1.23.2 h1:xmq9QRMWL8HTJyhAUBXy8FqIIQCYESeKfJL4DoGKiWQ= +gorm.io/gorm v1.23.2/go.mod h1:l2lP/RyAtc1ynaTjFksBde/O8v9oOGIApu2/xRitmZk= 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= diff --git a/models/cost.go b/models/cost.go new file mode 100644 index 0000000..ada1feb --- /dev/null +++ b/models/cost.go @@ -0,0 +1,40 @@ +package models + +import "gorm.io/gorm" + +type CPUCost struct { + gorm.Model + Value float64 `gorm:"not null"` + TenantId int32 `gorm:"not null"` +} + +type MemoryCost struct { + gorm.Model + Value float64 `gorm:"not null"` + TenantId int32 `gorm:"not null"` +} + +type IngressCost struct { + gorm.Model + Value float64 `gorm:"not null"` + TenantId int32 `gorm:"not null"` +} + +type StorageCost struct { + gorm.Model + Value float64 `gorm:"not null"` + TenantId int32 `gorm:"not null"` + StorageClass string `gorm:"not null"` +} + +type MonthlyCost struct { + gorm.Model + Month int32 `gorm:"not null"` + Year int32 `gorm:"not null"` + TenantId int32 `gorm:"not null"` + CPUCost float64 `gorm:"not null"` + MemoryCost float64 `gorm:"not null"` + IngressCost float64 `gorm:"not null"` + StorageCost float64 `gorm:"not null"` + TotalCost float64 `gorm:"not null"` +} diff --git a/models/doc.go b/models/doc.go new file mode 100644 index 0000000..7165f40 --- /dev/null +++ b/models/doc.go @@ -0,0 +1,18 @@ +/* +Copyright 2022 Netrics AG + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +// Package controllers is responsible to set up the controllers for the http handlers +package models diff --git a/models/tenant.go b/models/tenant.go new file mode 100644 index 0000000..e900809 --- /dev/null +++ b/models/tenant.go @@ -0,0 +1,9 @@ +package models + +import "gorm.io/gorm" + +type Tenant struct { + gorm.Model + Id int32 `gorm:"primary_key"` + GitHubTeamSlug string `gorm:"not null;unique"` +} diff --git a/routes/doc.go b/routes/doc.go index 89b9c29..65b803f 100644 --- a/routes/doc.go +++ b/routes/doc.go @@ -1,5 +1,5 @@ /* -Copyright 2022 Jan Lauber +Copyright 2022 Netrics AG Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. diff --git a/routes/routes.go b/routes/routes.go index c21a6b1..36783c2 100644 --- a/routes/routes.go +++ b/routes/routes.go @@ -43,6 +43,7 @@ func Setup(app *fiber.App, clientset *kubernetes.Clientset) { costs.Get("/memory", controllers.GetMemoryCostSum) costs.Get("/storage", controllers.GetStorageCostSum) costs.Get("/ingress", controllers.GetIngressCostSum) + costs.Get("/monthly", controllers.GetMonthlyCostSum) // Quotas quotas := v1.Group(":tenant/quotas") diff --git a/tenant-api.go b/tenant-api.go index aab301a..818d70d 100644 --- a/tenant-api.go +++ b/tenant-api.go @@ -1,5 +1,5 @@ /* -Copyright 2022 Jan Lauber +Copyright 2022 Netrics AG Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. @@ -24,6 +24,7 @@ import ( "github.com/gofiber/fiber/v2/middleware/cors" "github.com/gofiber/fiber/v2/middleware/limiter" "github.com/gofiber/template/html" + "github.com/natron-io/tenant-api/database" "github.com/natron-io/tenant-api/routes" "github.com/natron-io/tenant-api/util" "github.com/slack-go/slack" @@ -58,11 +59,59 @@ func init() { util.ErrorLogger.Println("Error loading env variables") os.Exit(1) } + + if util.COST_PERSISTENCY { + if err := database.InitDB(); err != nil { + util.ErrorLogger.Println("Error initializing database") + os.Exit(1) + } + util.InfoLogger.Println("Database is connected") + restartCounter := 1 + + go func() { + for { + time.Sleep(time.Duration(util.COST_PERSISTENCY_INTERVAL) * time.Second) + util.InfoLogger.Println("Saving costs...") + err := util.SaveCostsToDB() + if err != nil { + util.ErrorLogger.Println("Error saving costs:", err) + if restartCounter <= 5 { + util.InfoLogger.Printf("Restarting database %d time(s)", restartCounter) + err := database.InitDB() + if err != nil { + util.ErrorLogger.Println("Error restarting database") + } + restartCounter++ + } else { + util.ErrorLogger.Println("Shutting down application") + os.Exit(1) + } + } else { + util.InfoLogger.Println("Costs saved") + } + } + }() + + go func() { + for { + // if last day of current month is reached, create monthly report cost + if time.Now().Day() == 5 { + util.InfoLogger.Println("Creating monthly report cost") + err := util.CreateMonthlyCostReport() + if err != nil { + util.ErrorLogger.Println("Error creating monthly report cost:", err) + } + } + time.Sleep(time.Hour * 24) + } + }() + + } } func main() { - engine := html.New("./views", ".html") + engine := html.New("./web/views", ".html") if util.SLACK_TOKEN != "" { util.SlackClient = slack.New(util.SLACK_TOKEN) @@ -86,8 +135,8 @@ func main() { }, })) - app.Static("/styles", "./static/styles") - app.Static("/images", "./static/images") + app.Static("/styles", "./web/static/styles") + app.Static("/images", "./web/static/images") app.Get("/", func(c *fiber.Ctx) error { // set header to html diff --git a/test/parse_test.go b/test/parse_test.go new file mode 100644 index 0000000..7516332 --- /dev/null +++ b/test/parse_test.go @@ -0,0 +1,17 @@ +package util + +import ( + "testing" + + "github.com/natron-io/tenant-api/util" +) + +func TestContains(t *testing.T) { + tenants := []string{"tenant1", "tenant2", "tenant3"} + if !util.Contains("tenant1", tenants) { + t.Error("tenant1 should be in the tenants slice") + } + if util.Contains("tenant4", tenants) { + t.Error("tenant4 should not be in the tenants slice") + } +} diff --git a/util/costs.go b/util/costs.go index c9f59e0..e9bf314 100644 --- a/util/costs.go +++ b/util/costs.go @@ -3,6 +3,10 @@ package util import ( "fmt" "strings" + "time" + + "github.com/natron-io/tenant-api/database" + "github.com/natron-io/tenant-api/models" ) var ( @@ -18,6 +22,11 @@ var ( INGRESS_DISCOUNT_PERCENT float64 ) +type costSum struct { + Sum float64 + Count int32 +} + // GetCPUCost returns the cost of the provided MiliCPU func GetCPUCost(millicores float64) float64 { // return per core @@ -74,6 +83,388 @@ func GetIngressCostByDomain(hostnameStrings []string) float64 { return tenantIngressCostsPerDomainSum } +// GetIngressCost returns the cost of the provided Ingress func GetIngressCost(ingressCount int) float64 { return INGRESS_COST * float64(ingressCount) * (1 - INGRESS_DISCOUNT_PERCENT) } + +// SaveCostsDB saves all the costs of a tenant in the database +func SaveCostsToDB() error { + var err error + var tenants []models.Tenant + var cpuCost models.CPUCost + var memoryCost models.MemoryCost + var ingressCost models.IngressCost + + db := database.DBConn + + // get all namespaces in the kubernetes cluster + namespaces, err := GetNamespaces() + if err != nil { + return err + } + + for _, namespace := range namespaces { + tenant := models.Tenant{ + GitHubTeamSlug: namespace, + } + + tenants = append(tenants, tenant) + + } + + // add or update the tenants in the database + for _, tenant := range tenants { + // get the tenant from the database + err = db.Where("git_hub_team_slug = ?", tenant.GitHubTeamSlug).First(&tenant).Error + if err != nil { + // if the tenant is not in the database, add it + err = db.Create(&tenant).Error + if err != nil { + return err + } + } + } + + // get all tenants from the database + err = db.Find(&tenants).Error + if err != nil { + return err + } + + tenantCPURequests, err := GetCPURequestsSumByTenant(namespaces) + if err != nil { + return err + } + + tenantMemoryRequests, err := GetMemoryRequestsSumByTenant(namespaces) + if err != nil { + return err + } + + tenantPVCs, err := GetStorageRequestsSumByTenant(namespaces) + if err != nil { + return err + } + + tenantsIngressRequests, err := GetIngressRequestsSumByTenant(namespaces) + if err != nil { + return err + } + + for _, tenant := range tenants { + cpuCost = models.CPUCost{ + TenantId: tenant.Id, + Value: GetCPUCost(float64(tenantCPURequests[tenant.GitHubTeamSlug])), + } + + err = db.Create(&cpuCost).Error + if err != nil { + return err + } + + memoryCost = models.MemoryCost{ + TenantId: tenant.Id, + Value: GetMemoryCost(float64(tenantMemoryRequests[tenant.GitHubTeamSlug])), + } + + err = db.Create(&memoryCost).Error + if err != nil { + return err + } + + for storageClass, pvc := range tenantPVCs[tenant.GitHubTeamSlug] { + storageClassCost, err := GetStorageCost(storageClass, float64(pvc)) + if err != nil { + return err + } + storageCost := models.StorageCost{ + TenantId: tenant.Id, + StorageClass: storageClass, + Value: storageClassCost, + } + + err = db.Create(&storageCost).Error + if err != nil { + return err + } + } + + if !INGRESS_COST_PER_DOMAIN { + ingressCost = models.IngressCost{ + TenantId: tenant.Id, + Value: GetIngressCost(len(tenantsIngressRequests[tenant.GitHubTeamSlug])), + } + + err = db.Create(&ingressCost).Error + if err != nil { + return err + } + InfoLogger.Printf("Ingress cost for tenant %s: %f", tenant.GitHubTeamSlug, ingressCost.Value) + } else { + ingressCost = models.IngressCost{ + TenantId: tenant.Id, + Value: GetIngressCostByDomain(tenantsIngressRequests[tenant.GitHubTeamSlug]), + } + + err = db.Create(&ingressCost).Error + if err != nil { + return err + } + InfoLogger.Printf("Ingress cost for tenant %s: %f", tenant.GitHubTeamSlug, ingressCost.Value) + } + + } + + return nil +} + +// +func GetCPUCostsByMonthAndTenant(tenant models.Tenant) (models.CPUCost, error) { + + db := database.DBConn + + // get all costs for the tenant for the current month + var costs []models.CPUCost + err := db.Where("tenant_id = ? AND created_at BETWEEN ? AND ?", tenant.Id, time.Now().AddDate(0, -1, 0), time.Now()).Find(&costs).Error + if err != nil { + return models.CPUCost{}, err + } + + // calculate average cost for the month + var sum float64 + var count int + var averageCost float64 + for _, cost := range costs { + sum += cost.Value + count++ + } + if count == 0 { + averageCost = 0 + } else { + averageCost = sum / float64(count) + } + + cpuCosts := models.CPUCost{ + TenantId: tenant.Id, + Value: averageCost, + } + + return cpuCosts, nil +} + +func GetMemoryCostsByMonthAndTenant(tenant models.Tenant) (models.MemoryCost, error) { + + db := database.DBConn + + // get every cost for the tenant for the current month day 0 until now + var costs []models.MemoryCost + err := db.Where("tenant_id = ? AND created_at BETWEEN ? AND ?", tenant.Id, time.Now().AddDate(0, -1, 0), time.Now()).Find(&costs).Error + if err != nil { + return models.MemoryCost{}, err + } + + // calculate average cost for the month + var sum float64 + var count int + var averageCost float64 + + for _, cost := range costs { + sum += cost.Value + count++ + } + if count == 0 { + averageCost = 0 + } else { + averageCost = sum / float64(count) + } + + memoryCosts := models.MemoryCost{ + TenantId: tenant.Id, + Value: averageCost, + } + + return memoryCosts, nil +} + +func GetStorageCostsByMonthAndTenant(tenant models.Tenant) ([]models.StorageCost, error) { + + storageCostsTemp := make(map[string]costSum) + storageCosts := make([]models.StorageCost, 0) + + db := database.DBConn + + // get all costs for the tenant for the current month + var costs []models.StorageCost + + err := db.Where("tenant_id = ? AND created_at BETWEEN ? AND ?", tenant.Id, time.Now().AddDate(0, -1, 0), time.Now()).Find(&costs).Error + if err != nil { + return nil, err + } + + // calculate average cost for the month for each storage class + var averageCost float64 + + for _, cost := range costs { + storageCostsTemp[cost.StorageClass] = costSum{ + Sum: storageCostsTemp[cost.StorageClass].Sum + cost.Value, + Count: storageCostsTemp[cost.StorageClass].Count + 1, + } + } + + for storageClass, cost := range storageCostsTemp { + averageCost = cost.Sum / float64(cost.Count) + storageCosts = append(storageCosts, models.StorageCost{ + TenantId: tenant.Id, + StorageClass: storageClass, + Value: averageCost, + }) + } + + return storageCosts, nil +} + +func GetIngressCostsByMonthAndTenant(tenant models.Tenant) (models.IngressCost, error) { + + db := database.DBConn + + // get all costs for the tenant for the current month + var costs []models.IngressCost + err := db.Where("tenant_id = ? AND created_at BETWEEN ? AND ?", tenant.Id, time.Now().AddDate(0, -1, 0), time.Now()).Find(&costs).Error + if err != nil { + return models.IngressCost{}, err + } + + // calculate average cost for the month + var sum float64 + var count int + var averageCost float64 + + for _, cost := range costs { + sum += cost.Value + count++ + } + + if count == 0 { + averageCost = 0 + } else { + averageCost = sum / float64(count) + } + + ingressCost := models.IngressCost{ + TenantId: tenant.Id, + Value: averageCost, + } + + return ingressCost, nil + +} + +func CreateMonthlyCostReport() error { + var err error + var tenants []models.Tenant + var monthlyCosts []models.MonthlyCost + + db := database.DBConn + + // get all namespaces in the kubernetes cluster + namespaces, err := GetNamespaces() + if err != nil { + return err + } + + for _, namespace := range namespaces { + tenant := models.Tenant{ + GitHubTeamSlug: namespace, + } + + tenants = append(tenants, tenant) + + } + + // add or update the tenants in the database + for _, tenant := range tenants { + // get the tenant from the database + err = db.Where("git_hub_team_slug = ?", tenant.GitHubTeamSlug).First(&tenant).Error + if err != nil { + // if the tenant is not in the database, add it + err = db.Create(&tenant).Error + if err != nil { + return err + } + } + } + + // get all tenants from the database + err = db.Find(&tenants).Error + if err != nil { + return err + } + + for _, tenant := range tenants { + cpuCosts, err := GetCPUCostsByMonthAndTenant(tenant) + if err != nil { + return err + } + + memoryCosts, err := GetMemoryCostsByMonthAndTenant(tenant) + if err != nil { + return err + } + + storageCosts, err := GetStorageCostsByMonthAndTenant(tenant) + if err != nil { + return err + } + var storageClassCostSum float64 + for _, storageClass := range storageCosts { + storageClassCostSum = storageClassCostSum + storageClass.Value + } + + ingressCosts, err := GetIngressCostsByMonthAndTenant(tenant) + if err != nil { + return err + } + + // get last month and year + lastMonth := time.Now().AddDate(0, -1, 0) + lastYear := lastMonth.Year() + + // convert month and year to int + monthInt := int32(lastMonth.Month()) + yearInt := int32(lastYear) + + monthlyCosts = append(monthlyCosts, models.MonthlyCost{ + TenantId: tenant.Id, + CPUCost: cpuCosts.Value, + MemoryCost: memoryCosts.Value, + StorageCost: storageClassCostSum, + IngressCost: ingressCosts.Value, + Month: monthInt, + Year: yearInt, + TotalCost: cpuCosts.Value + memoryCosts.Value + storageClassCostSum + ingressCosts.Value, + }) + } + + // add or update the monthly costs in the database + for _, monthlyCost := range monthlyCosts { + // get the monthly cost from the database + err = db.Where("tenant_id = ? AND month = ? AND year = ?", monthlyCost.TenantId, monthlyCost.Month, monthlyCost.Year).First(&monthlyCost).Error + if err != nil { + // if the monthly cost is not in the database, add it + err = db.Create(&monthlyCost).Error + if err != nil { + return err + } + + } else { + // if the monthly cost is in the database, update it + err = db.Save(&monthlyCost).Error + if err != nil { + return err + } + } + } + + return nil +} diff --git a/util/doc.go b/util/doc.go index 610ea5f..8431216 100644 --- a/util/doc.go +++ b/util/doc.go @@ -1,5 +1,5 @@ /* -Copyright 2022 Jan Lauber +Copyright 2022 Netrics AG Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. diff --git a/util/os.go b/util/env.go similarity index 66% rename from util/os.go rename to util/env.go index e5e37cd..fa27553 100644 --- a/util/os.go +++ b/util/env.go @@ -5,11 +5,16 @@ import ( "os" "strconv" "strings" + + "github.com/natron-io/tenant-api/database" ) var ( - err error - CORS string + err error + CORS string + COST_PERSISTENCY bool + COST_PERSISTENCY_INTERVAL int + DEBUG bool ) // LoadEnv loads OS environment variables @@ -158,7 +163,7 @@ func LoadEnv() error { storageClassesInCluster, err := GetStorageClassesInCluster() if err != nil { - err = errors.New("cannot get storage classes in cluster") + err = errors.New("cannot get storage classes in cluster " + err.Error()) ErrorLogger.Println(err) Status = "Error: " + err.Error() os.Exit(1) @@ -182,5 +187,79 @@ func LoadEnv() error { InfoLogger.Printf("cost for storage class default set using default: %f", STORAGE_COST["default"]["cost"]) } + // ============= // + // Database // + // ============= // + if COST_PERSISTENCY, err = strconv.ParseBool(os.Getenv("COST_PERSISTENCY")); !COST_PERSISTENCY || err != nil { + WarningLogger.Println("COST_PERSISTENCY is not set or invalid bool value") + COST_PERSISTENCY = false + InfoLogger.Printf("COST_PERSISTENCY set using default: %t", COST_PERSISTENCY) + } else { + InfoLogger.Printf("COST_PERSISTENCY set using env: %t", COST_PERSISTENCY) + } + + if COST_PERSISTENCY_INTERVAL, err = strconv.Atoi(os.Getenv("COST_PERSISTENCY_INTERVAL")); COST_PERSISTENCY_INTERVAL == 0 || err != nil { + WarningLogger.Println("COST_PERSISTENCY_INTERVAL is not set or invalid int value") + COST_PERSISTENCY_INTERVAL = 3600 + InfoLogger.Printf("COST_PERSISTENCY_INTERVAL set using default (1h): %d", COST_PERSISTENCY_INTERVAL) + } else { + InfoLogger.Printf("COST_PERSISTENCY_INTERVAL set using env: %d", COST_PERSISTENCY_INTERVAL) + } + + if database.DB_HOST = os.Getenv("DB_HOST"); database.DB_HOST == "" { + WarningLogger.Println("DB_HOST is not set") + database.DB_HOST = "localhost" + InfoLogger.Printf("DB_HOST set using default: %s", database.DB_HOST) + } else { + InfoLogger.Printf("DB_HOST set using env: %s", database.DB_HOST) + } + + if database.DB_PORT = os.Getenv("DB_PORT"); database.DB_PORT == "" { + WarningLogger.Println("DB_PORT is not set") + database.DB_PORT = "5432" + InfoLogger.Printf("DB_PORT set using default: %s", database.DB_PORT) + } else { + InfoLogger.Printf("DB_PORT set using env: %s", database.DB_PORT) + } + + if database.DB_USER = os.Getenv("DB_USER"); database.DB_USER == "" { + WarningLogger.Println("DB_USER is not set") + database.DB_USER = "postgres" + InfoLogger.Printf("DB_USER set using default: %s", database.DB_USER) + } else { + InfoLogger.Printf("DB_USER set using env: %s", database.DB_USER) + } + + if database.DB_PASSWORD = os.Getenv("DB_PASSWORD"); database.DB_PASSWORD == "" { + WarningLogger.Println("DB_PASSWORD is not set") + database.DB_PASSWORD = "postgres" + InfoLogger.Printf("DB_PASSWORD set using default: %s", database.DB_PASSWORD) + } else { + InfoLogger.Printf("DB_PASSWORD set using env: %s", database.DB_PASSWORD) + } + + if database.DB_NAME = os.Getenv("DB_NAME"); database.DB_NAME == "" { + WarningLogger.Println("DB_NAME is not set") + database.DB_NAME = "postgres" + InfoLogger.Printf("DB_NAME set using default: %s", database.DB_NAME) + } else { + InfoLogger.Printf("DB_NAME set using env: %s", database.DB_NAME) + } + + if database.DB_SSLMODE = os.Getenv("DB_SSLMODE"); database.DB_SSLMODE == "" { + WarningLogger.Println("DB_SSLMODE is not set") + database.DB_SSLMODE = "disable" + InfoLogger.Printf("DB_SSLMODE set using default: %s", database.DB_SSLMODE) + } else { + InfoLogger.Printf("DB_SSLMODE set using env: %s", database.DB_SSLMODE) + } + + if DEBUG, err = strconv.ParseBool(os.Getenv("DEBUG")); DEBUG || err != nil { + InfoLogger.Printf("DEBUG set using env: %t", DEBUG) + } else { + InfoLogger.Printf("DEBUG set using default: %t", DEBUG) + DEBUG = false + } + return nil } diff --git a/util/k8s.go b/util/k8s.go index a4f3db6..8213a04 100644 --- a/util/k8s.go +++ b/util/k8s.go @@ -189,9 +189,11 @@ func GetIngressRequestsSumByTenant(tenants []string) (map[string][]string, error continue } - // apend ingress hostname to the list of ingress for the tenant + // apend ingress hostname to the list of ingress for the tenant if it doesn't already exist for _, rule := range ingress.Spec.Rules { - tenantsIngress[tenant] = append(tenantsIngress[tenant], rule.Host) + if !contains(tenantsIngress[tenant], rule.Host) { + tenantsIngress[tenant] = append(tenantsIngress[tenant], rule.Host) + } } } } @@ -199,6 +201,15 @@ func GetIngressRequestsSumByTenant(tenants []string) (map[string][]string, error return tenantsIngress, nil } +func contains(s1 []string, s2 string) bool { + for _, v := range s1 { + if v == s2 { + return true + } + } + return false +} + func GetStorageClassesInCluster() ([]string, error) { storageClasses := make([]string, 0) scList, err := Clientset.StorageV1().StorageClasses().List(context.TODO(), metav1.ListOptions{}) @@ -213,3 +224,18 @@ func GetStorageClassesInCluster() ([]string, error) { return storageClasses, nil } + +func GetNamespaces() ([]string, error) { + namespaces := make([]string, 0) + namespaceList, err := Clientset.CoreV1().Namespaces().List(context.TODO(), metav1.ListOptions{}) + + if err != nil && !strings.Contains(err.Error(), "not found") { + return nil, err + } + + for _, namespace := range namespaceList.Items { + namespaces = append(namespaces, namespace.Name) + } + + return namespaces, nil +} diff --git a/util/parse.go b/util/parse.go index feac7e4..84b2494 100644 --- a/util/parse.go +++ b/util/parse.go @@ -7,5 +7,9 @@ func Contains(tenant string, tenants []string) bool { return true } } + if DEBUG { + InfoLogger.Printf("tenant %s is not in the list of tenants %v", tenant, tenants) + return true + } return false } diff --git a/static/images/GitHub-Mark-Light-32px.png b/web/static/images/GitHub-Mark-Light-32px.png similarity index 100% rename from static/images/GitHub-Mark-Light-32px.png rename to web/static/images/GitHub-Mark-Light-32px.png diff --git a/static/styles/bootstrap-social.min.css b/web/static/styles/bootstrap-social.min.css similarity index 100% rename from static/styles/bootstrap-social.min.css rename to web/static/styles/bootstrap-social.min.css diff --git a/static/styles/bootstrap.min.css b/web/static/styles/bootstrap.min.css similarity index 100% rename from static/styles/bootstrap.min.css rename to web/static/styles/bootstrap.min.css diff --git a/static/styles/main.css b/web/static/styles/main.css similarity index 100% rename from static/styles/main.css rename to web/static/styles/main.css diff --git a/views/index.html b/web/views/index.html similarity index 100% rename from views/index.html rename to web/views/index.html