diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 0000000..e52b48d --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,26 @@ +version: 2 +registries: + github: + type: git + url: https://github.com + username: ${{secrets.EXTERNAL_GITHUB_USER}} + password: ${{secrets.EXTERNAL_GITHUB_TOKEN}} + +updates: + - package-ecosystem: "gomod" + directory: "/" + registries: + - github + schedule: + interval: "daily" + commit-message: + prefix: chore + include: scope + - package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "daily" + commit-message: + prefix: chore + include: scope + diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml new file mode 100644 index 0000000..0683287 --- /dev/null +++ b/.github/workflows/build.yaml @@ -0,0 +1,45 @@ +name: build +on: [push, pull_request, merge_group] + +jobs: + build: + permissions: + contents: read + packages: write + runs-on: ubuntu-latest + steps: + - name: Set up Docker context for Buildx + id: buildx-context + run: | + docker context create builders + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@v3 + with: + version: latest + endpoint: builders + + - uses: docker/login-action@v3 + with: + registry: ghcr.io + username: ${{ github.actor }} + password: ${{ secrets.GITHUB_TOKEN }} + + - uses: actions/checkout@v4 + + - name: Build images + id: buildimages + run: | + img=ghcr.io/${{github.repository}}:${{github.sha}} + echo "IMAGE_NAME=${img} >> $GITHUB_OUTPUT" + docker pull ${img} || ( + docker buildx build --push --cache-to type=gha,mode=max --cache-from type=gha --progress plain --platform linux/arm64/v8,linux/amd64 --build-arg BUILDKIT_INLINE_CACHE=1 -t ${img} . + ) + + - name: Build main image images + id: buildmainimage + if: ${{ github.ref == 'refs/heads/main' && github.event_name != 'pull_request' }} + run: | + img=ghcr.io/${{github.repository}}:latest + echo "IMAGE_NAME=${img} >> $GITHUB_OUTPUT" + docker buildx build --push --cache-to type=gha,mode=max --cache-from type=gha --progress plain --platform linux/arm64/v8,linux/amd64 --build-arg BUILDKIT_INLINE_CACHE=1 -t ${img} . diff --git a/.github/workflows/golangci-lint.yml b/.github/workflows/golangci-lint.yml new file mode 100644 index 0000000..e40e495 --- /dev/null +++ b/.github/workflows/golangci-lint.yml @@ -0,0 +1,27 @@ +name: golangci-lint +on: + push: + branches: + - main + - master + pull_request: + merge_group: + +permissions: + contents: read + # Optional: allow read access to pull request. Use with `only-new-issues` option. + # pull-requests: read + +jobs: + golangci: + name: lint + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-go@v5 + with: + go-version: '1.22.4' + - name: golangci-lint + uses: golangci/golangci-lint-action@v6 + with: + version: v1.60.3 diff --git a/.github/workflows/manage-stale-branches.yml b/.github/workflows/manage-stale-branches.yml new file mode 100644 index 0000000..ccd7f44 --- /dev/null +++ b/.github/workflows/manage-stale-branches.yml @@ -0,0 +1,21 @@ +name: 'Manage stale branches' +on: + schedule: + # Run every monday at 10 am + - cron: "0 10 * * 1" + +jobs: + manage-stale-branches: + name: "Manage stale branches" + runs-on: ubuntu-latest + steps: + - name: "Manage Stale Branches" + id: manage-stale-branches + uses: crazy-matt/manage-stale-branches@1.1.1 + with: + gh_token: ${{ secrets.GITHUB_TOKEN }} + stale_older_than: 60 + suggestions_older_than: 30 + excluded_branches: | + origin/main + origin/master diff --git a/.github/workflows/test.yaml b/.github/workflows/test.yaml new file mode 100644 index 0000000..74c9fa3 --- /dev/null +++ b/.github/workflows/test.yaml @@ -0,0 +1,66 @@ +name: run tests + +on: + push: + pull_request: + merge_group: + +jobs: + testGo: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-go@v5 + with: + go-version: 1.22.4 + - name: Run CI script + run: | + go test -v -cover -covermode=count ./... + + # testGCloudIntegration: + # runs-on: ubuntu-latest + # steps: + # - uses: actions/checkout@v4 + # - uses: actions/setup-go@v5 + # with: + # go-version: 1.22.4 + # - name: Run CI script + # run: | + # go test -v ./pkg/grafanacloud/... + # env: + # GRAFANA_CLOUD_TOKEN: ${{ secrets.GRAFANA_CLOUD_TOKEN }} + + alloy_lint: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-go@v5 + with: + go-version: '1.22.4' + - run: | + cd pkg/controllers + curl -LO https://github.com/grafana/alloy/releases/download/v1.3.1/alloy-linux-amd64.zip && unzip alloy-linux-amd64.zip + go test -v ./... -args -verify-alloy + + testHelm: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + with: + depth: 0 + + - uses: azure/setup-helm@v4.2.0 + with: + version: v3.12.0 + + - name: Set up chart-testing + uses: helm/chart-testing-action@v2.6.1 + + - run: | + git fetch origin "${{ github.base_ref }}" + # Ensure the chart meets requirements + ct lint --remote origin --target-branch "${{ github.base_ref }}" --charts ./helm-chart/observability-operator + # Ensure the chart can be rendered with default values set and the generated yaml is coherent + helm template ./helm-chart/observability-operator + # Ensure the chart can be rendered with all values set and the generated yaml is coherent + helm template ./helm-chart/observability-operator -f ./helm-chart/observability-operator/values-test-full.yaml diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..51e3a7a --- /dev/null +++ b/.gitignore @@ -0,0 +1,28 @@ + +# Binaries for programs and plugins +*.exe +*.exe~ +*.dll +*.so +*.dylib + +# Test binary, build with `go test -c` +*.test + +# Output of the go coverage tool, specifically when used with LiteIDE +*.out + +# Kubernetes Generated files - skip generated files, except for vendored files + +!vendor/**/zz_generated.* + +# editor and IDE paraphernalia +.idea +*.swp +*.swo +*~ +.helm +vendor +profile.cov +junit.xml +.vscode \ No newline at end of file diff --git a/.golangci.yaml b/.golangci.yaml new file mode 100644 index 0000000..f591fef --- /dev/null +++ b/.golangci.yaml @@ -0,0 +1,58 @@ +# Options for analysis running. +run: + # Timeout for analysis, e.g. 30s, 5m. + # Default: 1m + timeout: 10m + # Exit code when at least one issue was found. + # Default: 1 + issues-exit-code: 2 + # If set, we pass it to "go list -mod={option}". From "go help modules": + # If invoked with -mod=readonly, the go command is disallowed from the implicit + # automatic updating of go.mod described above. Instead, it fails when any changes + # to go.mod are needed. This setting is most useful to check that go.mod does + # not need updates, such as in a continuous integration and testing system. + # If invoked with -mod=vendor, the go command assumes that the vendor + # directory holds the correct copies of dependencies and ignores + # the dependency descriptions in go.mod. + # + # Allowed values: readonly|vendor|mod + # Default: "" + modules-download-mode: readonly + +# output configuration options +output: + # The formats used to render issues. + # Formats: + # - `colored-line-number` + # - `line-number` + # - `json` + # - `colored-tab` + # - `tab` + # - `html` + # - `checkstyle` + # - `code-climate` + # - `junit-xml` + # - `github-actions` + # - `teamcity` + # - `sarif` + # Output path can be either `stdout`, `stderr` or path to the file to write to. + # + # For the CLI flag (`--out-format`), multiple formats can be specified by separating them by comma. + # The output can be specified for each of them by separating format name and path by colon symbol. + # Example: "--out-format=checkstyle:report.xml,json:stdout,colored-line-number" + # The CLI flag (`--out-format`) override the configuration file. + # + # Default: + # formats: + # - format: colored-line-number + # path: stdout + formats: + - format: colored-line-number + path: stdout + # Show statistics per linter. + # Default: false + show-stats: true +issues: + # Show only new issues created after git revision `REV`. + # Default: "" + new-from-rev: master diff --git a/.yamllint.yaml b/.yamllint.yaml new file mode 100644 index 0000000..1fd3784 --- /dev/null +++ b/.yamllint.yaml @@ -0,0 +1,7 @@ +extends: default + +rules: + line-length: disable + document-start: disable + indentation: + indent-sequences: consistent diff --git a/CODE-CONTRIBUTIONS.md b/CODE-CONTRIBUTIONS.md new file mode 100644 index 0000000..6cee11e --- /dev/null +++ b/CODE-CONTRIBUTIONS.md @@ -0,0 +1,280 @@ +## Pull Requests + +Code Review is an important part of any modern software development team. In this document we aim to give a set of guidelines and +_rules of thumb_. +These are not hard requirements for _every_ Pull Request, but they should help us making better decisions. + +### Why do we do Pull Requests (PRs)? + +We use PRs mainly for three reasons to: + +1. **Share knowledge**. Although this is not the only way we can achieve it, PullRequests are + a great helper to share the experiences, successes and failures, as well as the domain knowledge across + users and maintainers. +2. Overall, **improve our code and architecture quality**. +3. **Help catch problems** before they reach production. + +### Rules of thumb + +#### Issuer + +##### Avoid internal tools references + +This project is an open-source project and we should avoid references to internal tools, such as JIRA, Confluence, etc. +On a general manner, any reference must be publicly accessible so anyone can understand the context of the PR. +It is accepted to create references to resources that requires a login but allow access and account creation for anybody. + +If you feel the need to reference an internal tool, make sure you provide a good summary of the concerned ticket or document. +and use the word _internal_ to help identify private accesses for public readers. + +##### Link to the issue + +Any PR solving an existing GitHub issue must reference it. This will help to understand the context of the PR and the problem. +It will also help to track the changes and the history of the issue. + +##### Use correct PR title + +Any PR title should follow the next principles: + +* If it's a bugfix, the PR title should start with "fix:" +* If it's a library update, the PR title should start with "chore:" + +Good PR titles will help to understand what has changed over time +and will help to troubleshoot any issue in the future. +Good commit messages are particularly useful used combined +with `git log --graph --pretty=tformat:'%h %s'`. + +Format: + +```plaintext + +``` + +Example: + +```plaintext +fix: solve a race condition in the HTTP client that caused the app to crash +``` + +##### Make small Pull Requests + +Small Pull-Requests make the reviewers' lives easier and they tend to be faster to merge. There are a few options when it comes to +write small PRs: + +* Separate refactors from the actual task. You can do them either before or after. But if a refactor could help to a feature +* include it in the description, so you can get the context of the origin of the it. +* Separate _vendor_ code either in another _commit_ or in another PR. +* Create small PRs against a main-task Pull-Request. Another option could be to use +* [Maiao](https://github.com/adevinta/maiao), when you have write accesses to the code repository. +* Use [feature toggles](https://martinfowler.com/articles/feature-toggles.html) if you don't want the new code to run in + production just yet. + +**ProTip™**: To evaluate whether you should do one or multiple PRs, you may ask yourself _If I revert feature "add the ability to +count to 2" should "bump testing component version" also be reverted?_ + +##### Explain the _why_ and write good commit messages + +Explain **why** you are introducing these changes. For people with plenty of domain knowledge, it serves as a quick validation and +they will verify if the code does what you're saying it does, speeding up the review process. +For people with less domain knowledge — such as recent team members or people not familiar with the codebase - it will allow to do +a decent review. +Having people who are not that familiar with the codebase can both bring in a new perspective and it is also an efficient way to +share knowledge. + +**ProTip™**: If you write the _why_ in the commit message and have a single commit in the PR, GitHub will propagate that into the +PR description upon creation. + +**ProTip™**: If you write the _why_ in the commit message and use [maiao](https://github.com/adevinta/maiao), the PR +description will automatically be maintained from the commit message upon updates. +Here's an [example](https://github.com/adevinta/noe/pull/84). + +There are [plenty](https://chris.beams.io/posts/git-commit/) of +[resources](https://www.freecodecamp.org/news/writing-good-commit-messages-a-practical-guide/) on how to write good commit +messages. + +##### Link github issues + +If you're working on a feature or a bug, it's a good practice to +[link the issue in the PR](https://docs.github.com/en/issues/tracking-your-work-with-issues/linking-a-pull-request-to-an-issue#linking-a-pull-request-to-an-issue-using-a-keyword) +description. +This will help to understand the context of the PR and the problem. +It will also help to track the changes and the history of the issue and automate the closing of the issue upon PR merge. + +Example: + +```plaintext +Closes: #10 +``` + +##### Define non-goals + +One important note, aside from stating what's the _goal_ of the PR might be to state what's a _non-goal_. An example could be: + +> Non-Goals: Although I moved function A of place, this PR doesn't address a refactor because it would broaden the scope too +much. + +In this way, we're telling our reviewers that, albeit something was considered, it should not be part of the review, since it's +out of the scope. + +##### Be open to changes + +Finishing our tasks feels great. It liberates dopamine and gives us a sense of accomplishment. Nevertheless, there's a reason why +we don't commit directly into the _main_ branch and why we ask our peers to review our work: we want to learn from them and we +want to listen to different opinions. Don't get too attached to your code (It could help to think in terms of our code or team's +code, rather than my code) + +**Rule of thumb**: If you're not strongly opinionated and making the change is less effort than discussing it, we can apply it +directly and move on. + +##### You're the shepherd of the PR + +If a PR is getting too long to get reviewed, you should be the one contacting people that made comments/suggestions and are not +approving them, to make sure we move it forward. +If you don't reach consensus/commitment, you can involve the Tech Lead to help on solving the conflict. + +#### Reviewer + +##### Focus on the big picture + +We all have opinions and software — especially if it runs on the cloud — is never in a finished state. Consider the work and +effort the PR issuer already put in, +weigh how important the task is and how it fits in the bigger picture. If there are ten flaws in a PR, perhaps only three are +critical and/or important. +Therefore, the other can be marked as _optional_ or _nit_, so we don't overwhelm the PR issuer. + +##### Be clear about your requests + +State clearly what you're trying to achieve: are you blocking the PR? are you nitpicking? Is a question about the code that you +don't understand? Be explicit about it. + +If you think that a piece of code could be done differently, ask the following questions to yourself: + +* Does it really change what we are trying to achieve here? Is it worth to tackle this _now_? +* How many other recommendations did the issuer already accepted? Remember: there's so much we can do. Everything can be improved +* Do you have an alternative? Maybe you can put it in the comment or even use + [suggested changes](https://docs.github.com/en/github/collaborating-with-issues-and-pull-requests/incorporating-feedback-in-your-pull-request#applying-a-suggested-change), + to lower the effort of accepting it +* Is it a big change? Maybe it makes sense to check first — face-to-face, through a VC or Slack — with the PR issuer if it makes sense +* Is it a big change? Maybe it makes sense to check first — face-to-face, through a VC or Slack — with the PR issuer if it makes + sense + +**ProTip™**: aside of Github's Approve/Request Changes/Comment system, it can be useful to point if a comment is a _nit_, so we +inform the PR issuer that we don't expect it to land in the final code. + +##### Use commit suggestions + +If you have a clear idea how to improve a certain piece of code, you can use +[suggested changes](https://docs.github.com/en/github/collaborating-with-issues-and-pull-requests/incorporating-feedback-in-your-pull-request#applying-a-suggested-change) +and make the issuer's life easier. + +##### Not all suggestions will be merged + +And that's fine. Sometimes, it's just too much. Certainly, the comment you left will be taken into account the next time. + +### A word on kindness + +When doing a code review, we are not seeing each other most of the time, but rather each other's code. We can’t leverage body +language or happy, in-person smiles like office-dwelling teams. +This means close attention to the language and tone of our written feedback is crucial to team happiness and morale. + +Moreover, adopting an understanding tone will help your point of view to be understood, considered and achieve your goal to alter +the PR contents. + +**ProTip™**: [Grammarly](https://app.grammarly.com/) has an experimental feature to show the tone perceived by a text. Abusing it +to tune your message would help you find the right tone. + +#### Understand first + +You're reviewing someone's work and, usually, people don’t make mistakes intentionally. +Perhaps they were having a bad day, or couldn’t come up with a better solution. Take it easy. +Ask people why they did it like that, in a polite way. Sometimes you don’t understand the trade-offs that were made. +Sit down with them if needed. We should let our ego out of the door. + +##### Good 👍 ✅ + +> I see we're repeating this in a few similar classes instead of creating an abstraction for this use-case. +> Could you walk me through the changes so I get a better picture on this works? + +##### Bad 👎 ❌ + +> You should use a parent class here. DRY + +#### Be expressive + +Sometimes it’s difficult to convey meaning through text alone. Comments can come off as terse or rude, even when the reviewer +tried to convey them in a positive and helpful light. +Sprinkling in emojis can help elevate the voice and tone for many of our team members. It helps the author read the comment in the +reviewer's voice, as if they were delivering the feedback in person. + +##### Examples + +> This looks good. I think we can move forward. + +vs + +> Wooow! This looks good 👏 I think we can move forward and put this code in production! 🚀 + +#### Suggest, don't command + +People will be more enthusiastic about changing their work when you make polite suggestions instead of commanding actions. + +Having a commanding tone is more likely cause repulsion and close the conversation from the PR author, while adopting an +understanding tone is more likely to open the conversation and get your point of view to be considered. + +##### Good 👍 ✅ + +> Maybe it would make sense to rename the class `UserRepositoryImpl` to `RedisCachedUserRepository`, to improve readability. What +do you think? +> Perhaps I’m not understanding it correctly, but it seems that this code won’t work properly in X and Y cases. Can we review it? +If it has a bug we can fix it together! + +##### Bad 👎 ❌ + +> Change this! +> This has a bug here! Fix it and create a test + +#### Focus on the essential parts + +You shouldn’t expect that all of your comments end up in the code base. That’s totally OK. +If you think that you are requesting too much changes or that some of them are not that important, please refer to them as +recommendations for future work. People will still learn and will try to improve for the next time. + +It’s also possible that one or more engineers on the team have already pointed out numerous issues on a single pull request. While +the additional feedback would be technically helpful, it’s likely going to just "add to the pile" +and make the code author feel discouraged or overwhelmed. + +##### Good 👍 ✅ + +> [Future reference] This code looks good 🚀 In future occasions, we can also use it for comprehensions instead of `map()`. I +wouldn't change it now, though. +> [Nit] We could use XX instead of YY. + +##### Bad 👎 ❌ + +> We could use for comprehensions instead of `map()` +> Here too. +> Again, for comprehensions. +> Can you change XX for YY? + +#### Make someone's day + +Compliments are free to give. By leaving a sincere compliment or mention that you've learned something new you could be making +someone's day. + +##### Examples + +> I know we usually stub this out, but your approach of calling the actual method in this test is 💯. I feel much more confident +that we’ll catch changes to the API in the future. 😍 +> Looks like it's working as-is 🚀 Great work in this PR! Thanks for taking the time to write test cases for X and Y. I think we +should keep this spirit and improve our codebase a bit every day 👏👏👏 + +### References + +1. +2. +3. +4. +5. +6. +7. +8. diff --git a/CODEOWNERS b/CODEOWNERS new file mode 100644 index 0000000..fd0b9f7 --- /dev/null +++ b/CODEOWNERS @@ -0,0 +1,3 @@ +# Adding this CODEOWNERS file and adding another branch protection +# ` Require review from Code Owners ` we will ensure that other members can create branches (which means they can merge PRs) but only merge PRs that has been approved by Adevinta. +* @adevinta/cloud-platform-runtime \ No newline at end of file diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 0000000..f34571d --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,9 @@ +# Contributing + +Any contribution submitted for inclusion in the work by you shall be under +the terms and conditions of this license (MIT), without any additional +terms or conditions. Notwithstanding the above, nothing herein shall +supersede or modify the terms of any separate license agreement you may +have executed regarding your contributions. + +Please refer to the [code contribution guide](./CODE-CONTRIBUTIONS.md) diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..58500b9 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,23 @@ +# Build the manager binary +FROM golang:1.23.5 AS builder + +WORKDIR /workspace +# Copy the Go Modules manifests +COPY go.mod go.mod +COPY go.sum go.sum + +# Copy the go source +COPY cmd/observability-operator/main.go cmd/observability-operator/main.go +COPY pkg/ pkg/ + +# Build +RUN CGO_ENABLED=0 GOOS=linux GOARCH=amd64 GO111MODULE=on go build -a -o manager cmd/observability-operator/main.go + +# Use distroless as minimal base image to package the manager binary +# Refer to https://github.com/GoogleContainerTools/distroless for more details +FROM gcr.io/distroless/static:nonroot +WORKDIR / +COPY --from=builder /workspace/manager . +USER nonroot:nonroot + +ENTRYPOINT ["/manager"] diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..5ab934a --- /dev/null +++ b/LICENSE @@ -0,0 +1,22 @@ +MIT License + +Copyright (c) 2024 Adevinta + +Permission is hereby granted, free of charge, to any person obtaining +a copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to +permit persons to whom the Software is furnished to do so, subject to +the following conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE +LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION +OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION +WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..96c985d --- /dev/null +++ b/README.md @@ -0,0 +1,57 @@ +

+ + Quickstart  •   + Administrator docs  •   + Development docs  •   + +

+ +# Observability Operator + +Observability Operator is a Kubernetes Operator that manages and +orchestrates the infrastructure to collect and relay Observability +data to several destinations in multi-teant Kubernetes clusters. + +It aims to require minimal configuration and reduce the toil of a +feature team to get Observability data from their workloads by +providing sane defaults and automatically discovering as much +information as possible on behalf of the user. + +It captures telemetry of tenant workloads (won't instrument the entire +cluster), can capture metrics, logs and traces and will relay the +telemetry to the tenant systems. + +## Supported integrations + +* Metrics: scrape Prometheus format from Pods +* Logs: any format, preferably JSON +* Traces: OpenTelemetry format +* Destination: + * Metrics: any Prometheus RemoteWrite-compatible destination + * Logs: any Fluentd supported format + * Traces: any OpenTelemetry-compatible receiver + +## Technologies + +Observability Operator relies on the cluster already providing some +services. Cluster admins are expected to fulfill these needds in +advance: + +* [Prometheus Operator](https://github.com/prometheus-operator/prometheus-operator) +* [Kube Fluentd Operator](https://github.com/vmware/kube-fluentd-operator) + +For traces we use [Grafana +Alloy](https://grafana.com/docs/alloy/latest/) but we manage its +deployment directly in this operator, so there are no dependenciies +to install. + +## Contributing + +Refer to Please refer to the [code contribution guide](./CONTRIBUTING.md). + +You can find further details of the operator in our [development documentation](./docs/development/README.md). + +# License + +This project is released under MIT license. You can find a copy of the +license terms in the LICENSE file. diff --git a/cmd/observability-operator/main.go b/cmd/observability-operator/main.go new file mode 100644 index 0000000..02416df --- /dev/null +++ b/cmd/observability-operator/main.go @@ -0,0 +1,376 @@ +package main + +import ( + "flag" + "fmt" + "os" + "strings" + "time" + + advlog "github.com/adevinta/go-log-toolkit" + "github.com/adevinta/observability-operator/pkg/controllers" + "github.com/adevinta/observability-operator/pkg/grafanacloud" + "github.com/grafana/grafana-com-public-clients/go/gcom" + "github.com/sirupsen/logrus" + networkingv1 "k8s.io/api/networking/v1" + _ "k8s.io/client-go/plugin/pkg/client/auth/gcp" + "k8s.io/client-go/util/workqueue" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/cache" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/event" + metricsserver "sigs.k8s.io/controller-runtime/pkg/metrics/server" + + validatingfield "k8s.io/apimachinery/pkg/util/validation/field" + + apilabels "k8s.io/apimachinery/pkg/labels" + // +kubebuilder:scaffold:imports +) + +const ( + missing = "missing" +) + +type grafanaCloudClient interface { + controllers.GrafanaCloudClient + grafanacloud.GrafanaCloudStackLister +} + +func main() { + // ############## Manager options ############## + var metricsAddr string + flag.StringVar(&metricsAddr, "metrics-addr", ":8080", "The address the metric endpoint binds to.") + var enableLeaderElection bool + flag.BoolVar(&enableLeaderElection, "enable-leader-election", false, "Enable leader election for controller manager. Enabling this will ensure there is only one active controller manager.") + // ############################################# + + // ############## metadata to add to metrics, logs, traces... ############## + var clusterName string + flag.StringVar(&clusterName, "cluster-name", missing, "The name of the cluster being run. This will add the cluster tag to all collected metrics.") + var clusterRegion string + flag.StringVar(&clusterRegion, "cluster-region", missing, "The region the cluster runs in. This will add the region tag to all collected metrics.") + var clusterDomain string + flag.StringVar(&clusterDomain, "cluster-domain", "adevinta.com", "The domain used for labels, annotations and leader election.") + // ############################################# + + // ############## Feature toggles ############## + // Enable sending to GrafanaCloud if storage is defined as such + // TODO: Always enabled? Can we drop it for now? + var enableMetricsRemoteWrite bool + flag.BoolVar(&enableMetricsRemoteWrite, "metrics-remote-write-to-grafana-cloud", true, "Enable/Disable remote-write of metrics to Grafana Cloud") + // Enable VPA provisioning for Prometheus + var enableVpa bool + flag.BoolVar(&enableVpa, "enable-vpa", false, "Enable/Disable support for VPA") + // ############################################# + + // ############## Filtering/ignoring workloads ############## + // Avoids recognising apps with this label as valid to create + // Trace Collectors + var ignoreApps string + flag.StringVar(&ignoreApps, "exclude-apps-label", "", "comma-separated list of app label values to exclude, if any") + + // Filter workloads by label selector + var workloadLabelSelector string + flag.StringVar(&workloadLabelSelector, "exclude-workload-selector", "", "Exclude applications using the provided K8s Selector. It is a string specification of a equality-based or set-based selector as described in https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors . This selector will be matched against deployments, statefulsets, daemonsets, etc. Matching ones will be skipped") + // Filter namespaces by label selector + var namespaceLabelSelector string + flag.StringVar(&namespaceLabelSelector, "exclude-namespace-selector", "", "Exclude namespaces using the provided K8s Selector. It is a string specification of a equality-based or set-based selector as described in https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors . This selector will be matched against namespaces. Matching ones will be skipped") + + // Do not process workloads in these namespaces, nor react to + // its events + var ignoreNamespaces string + flag.StringVar(&ignoreNamespaces, "exclude-namespaces-name", "", "comma-separated list of namespaces to exclude, if any") + // ############################################# + + // ############## GrafanaCloud configuration ############## + // Name identifier of the GrafanaCloud organization controlling all stacks + var grafanaCloudOrgSlug string + flag.StringVar(&grafanaCloudOrgSlug, "grafana-cloud-organization-name", "adevinta", "The name of the grafanacloud organization hosting all stacks used as destination for telemetry") + // Name of the secret containing credentials to access GrafanaCloud + var grafanaCloudCredentials string + flag.StringVar(&grafanaCloudCredentials, "grafana-cloud-metrics-credentials", "", "Secret used to store Grafana Cloud credentials") + var grafanaCloudClientCache bool + flag.BoolVar(&grafanaCloudClientCache, "grafana-cloud-client-use-cache", false, "enables the usage of a client-level cache in the GrafanaCloud client. This can significantly reduce the amount of calls to the GCOM API for the stacks that exist.") + // Periodicity to reconcile the status of Grafana stacks for changes + var grafanaCloudStackReconcilePeriod string + defaultGrafanaCloudStackReconcilePeriod := 5 * time.Minute + flag.StringVar(&grafanaCloudStackReconcilePeriod, "grafana-cloud-stack-reconcile-period", defaultGrafanaCloudStackReconcilePeriod.String(), "Frequency to reconcile grafana stack change that triggers namespace reconcile (expressed in go duration: https://pkg.go.dev/time#ParseDuration)") + // ############################################# + + // ############## Prometheus configuration settings ############## + // Prometheus image details + var prometheusDockerImage string + flag.StringVar(&prometheusDockerImage, "prometheus-docker-image", "", "The name of the prometheus docker image to use.") + var prometheusDockerTag string + flag.StringVar(&prometheusDockerTag, "prometheus-docker-tag", "", "The prometheus version to use.") + // The name of the nodepool to target for Prometheus pods + // If empty, Prometheus pods do not target any nodepool + var prometheusNodeSelectorTarget string + flag.StringVar(&prometheusNodeSelectorTarget, "prometheus-node-selector-target", "", "Node-selector label target pool for Prometheus pods. If empty, no node-selector gets added.") + // Name of the namespace to hold Prometheus pods + var promNamespace string + flag.StringVar(&promNamespace, "prometheus-namespace", "platform-services", "Namespace in which to create prometheus instances") + // The service account to be used by the Prometheus instances, which provides the necessary permissions to scrape workloads + var prometheusServiceAccountName string + flag.StringVar(&prometheusServiceAccountName, "prometheus-service-account-name", "prometheus-tenant", "Service account to use for Prometheus") + // The pod priority class to use for Prometheus instances + var prometheusPodPriorityClassName string + flag.StringVar(&prometheusPodPriorityClassName, "prometheus-pod-priority-classname", "", "Prometheus pod priority class name to use for Prometheus pods") + var prometheusMonitoringTargetName string + flag.StringVar(&prometheusMonitoringTargetName, "prometheus-monitoring-target-name", "", "Name of the secret to store the monitoring target and extra scraping configuration") + var prometheusExtraExternalLabels string + flag.StringVar(&prometheusExtraExternalLabels, "prometheus-extra-external-labels", "", "Extra external labels to be added to the prometheus configuration. Format: key1:value1,key2:value2") + + // ############################################# + + // ############## Logs configuration settings ############## + // Namespace to locate the ConfigMap with FluentD's configuration + var fluentdLokiConfigMapNamespace string + flag.StringVar(&fluentdLokiConfigMapNamespace, "logs-fluentd-loki-configmap-namespace", "", "The namespace of the fluentd-loki configmap.") + // Name of the ConfigMap containing FluentD's configuration + var fluentdLokiConfigMapName string + flag.StringVar(&fluentdLokiConfigMapName, "logs-fluentd-loki-configmap-name", "", "The name of the fluentd-loki configmap.") + // Key used in the ConfigMap with FluentD's configuration that contains the configuration + var fluentdLokiConfigMapKey string + flag.StringVar(&fluentdLokiConfigMapKey, "logs-fluentd-loki-configmap-key", "", "The data key of the fluentd-loki configmap where the fluentd rules will be injected.") + // ############################################# + + // ############## Traces configuration settings ############## + // Name of the namespace to hold Alloy collectors + var tracesNamespace string + flag.StringVar(&tracesNamespace, "traces-namespace", "observability", "Namespace in which to create traces collector (alloy) instances") + // ############################################# + + // END of CLI options + flag.Parse() + + // ############## Parse extra external labels for the Prometheus instance ############## + prometheusExtraExternalLabelsMap := make(map[string]string) + if prometheusExtraExternalLabels != "" { + labels := strings.Split(prometheusExtraExternalLabels, ",") + for _, label := range labels { + kv := strings.Split(label, ":") + if len(kv) != 2 { + fmt.Println("invalid extra external label format") + flag.Usage() + os.Exit(1) + } + prometheusExtraExternalLabelsMap[kv[0]] = kv[1] + } + } + // ############################################# + + // ############## Initialize label names for configuration ############## + grafanacloud.Config = grafanacloud.NewConfig(clusterDomain) + controllers.Config = controllers.NewConfig(clusterDomain) + // ############################################# + + // ############## Verify configuration is valid ############## + grafanaCloudToken, tokenPresent := os.LookupEnv("GRAFANA_CLOUD_TOKEN") + if !tokenPresent { + fmt.Println("missing GRAFANA_CLOUD_TOKEN env variable") + os.Exit(1) + } + + grafanaCloudTracesToken, tokenPresent := os.LookupEnv("GRAFANA_CLOUD_TRACES_TOKEN") + if !tokenPresent { + fmt.Println("missing GRAFANA_CLOUD_TRACES_TOKEN env variable") + flag.Usage() + os.Exit(1) + } + + if clusterName == missing { + fmt.Println("missing cluster name") + flag.Usage() + os.Exit(1) + } + if clusterRegion == missing { + fmt.Println("missing cluster region") + flag.Usage() + os.Exit(1) + } + // ############################################# + + // ############## Set up logging facilities ############## + log := logrus.New() + log.SetFormatter(&logrus.JSONFormatter{}) + log.SetLevel(logrus.InfoLevel) + ctrl.SetLogger(advlog.NewLogr(log)) + setupLog := ctrl.Log.WithName("setup") + // ############################################# + + // ############## Set up the Manager ############## + scheme := controllers.NewScheme() + cfg := ctrl.GetConfigOrDie() + mgr, err := ctrl.NewManager(cfg, ctrl.Options{ + Scheme: scheme, + Metrics: metricsserver.Options{ + BindAddress: metricsAddr, + }, + LeaderElection: enableLeaderElection, + LeaderElectionID: "8cba130d.prometheus." + clusterDomain, + Cache: cache.Options{ + ByObject: map[client.Object]cache.ByObject{ + &networkingv1.NetworkPolicy{}: { + Namespaces: map[string]cache.Config{ + tracesNamespace: {}, + }, + }, + }, + }, + }) + if err != nil { + setupLog.Error(err, "unable to start manager") + os.Exit(1) + } + // ############################################# + + // ############## Verify we can connect to GrafanaCloud ############## + config := gcom.NewConfiguration() + config.AddDefaultHeader("Authorization", "Bearer "+grafanaCloudToken) + config.Host = "grafana.com" + config.Scheme = "https" + + gcomClient := gcom.NewAPIClient(config) + var gcClient grafanaCloudClient + gcClient = grafanacloud.NewClient(ctrl.Log.WithName("controllers").WithName("Grafana Cloud Client (non-cached)"), gcomClient, grafanaCloudOrgSlug) + + if grafanaCloudClientCache { + gcClient = grafanacloud.NewCachedClient(ctrl.Log.WithName("controllers").WithName("Grafana Cloud Client (cached)"), gcClient.(*grafanacloud.Client)) + } + + // ############################################# + + path := validatingfield.NewPath("metadata", "labels") + appSelector, err := apilabels.Parse(workloadLabelSelector, validatingfield.WithPath(path)) + if err != nil { + setupLog.Error(err, "unable to parse application selector", workloadLabelSelector) + os.Exit(1) + } + namespaceSelector, err := apilabels.Parse(namespaceLabelSelector, validatingfield.WithPath(path)) + if err != nil { + setupLog.Error(err, "unable to parse namespace selector", namespaceSelector) + os.Exit(1) + } + + // ############## Set up Pod Reconciler ############## + if err = (&controllers.PodReconciler{ + Client: mgr.GetClient(), + Log: ctrl.Log.WithName("controllers").WithName("Deployment"), + Scheme: mgr.GetScheme(), + ExcludeWorkloadLabelSelector: appSelector, + ExcludeNamespaceLabelSelector: namespaceSelector, + IgnoreApps: strings.Split(ignoreApps, ","), + IgnoreNamespaces: strings.Split(ignoreNamespaces, ","), + TracesNamespace: tracesNamespace, + GrafanaCloudClient: gcClient, + GrafanaCloudTracesToken: grafanaCloudTracesToken, + ClusterName: clusterName, + EnableVPA: enableVpa, + }).SetupWithManager(mgr); err != nil { + setupLog.Error(err, "unable to create controller", "controller", "Deployment") + os.Exit(1) + } + // ############################################# + + // ############## Set up GrafanaStackReconciler ############## + // we need a client that can manage resources only in a given namespace + // by default, clients generated by cluster-controller manager are generated + // for cluster scope controllers are using cluster-wide caches. + // In the case of the loki configuration, we want are restricting the role to manage a single resource. + // Hence we create a dedicated client for it. + // TODO: Consider using the general client now that we restrict the cache for specific objects + namespacedClient, err := client.New(ctrl.GetConfigOrDie(), client.Options{Scheme: scheme}) + if err != nil { + setupLog.Error(err, "unable to create uncached client", "controller", "Grafana Cloud Config Updater") + os.Exit(1) + } + + gcConfigUpdater := &grafanacloud.GrafanaCloudConfigUpdater{ + Log: ctrl.Log.WithName("controllers").WithName("Grafana Cloud Config Updater"), + Client: namespacedClient, + GrafanaCloudClient: gcClient, + ClusterName: clusterName, + ClusterRegion: clusterRegion, + ConfigMapNamespace: fluentdLokiConfigMapNamespace, + ConfigMapName: fluentdLokiConfigMapName, + ConfigMapLokiKey: fluentdLokiConfigMapKey, + } + + go gcConfigUpdater.Start(workqueue.NewTypedRateLimitingQueue(workqueue.NewTypedItemExponentialFailureRateLimiter[string](time.Millisecond, 30*time.Second))) + + grafanaStackChangeEvents := make(chan event.GenericEvent, 10) + stackReconcilePeriod, err := time.ParseDuration(grafanaCloudStackReconcilePeriod) + if err != nil { + setupLog.Error(err, "failed to parse grafanacloud reconciliation period. Using default %v ", defaultGrafanaCloudStackReconcilePeriod) + stackReconcilePeriod = defaultGrafanaCloudStackReconcilePeriod + } + + tick := time.NewTicker(stackReconcilePeriod) + + if err = (&grafanacloud.GrafanaStackReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("GrafanaStack"), + GrafanaCloudClient: gcClient, + GrafanaStackChangeEvents: grafanaStackChangeEvents, + }).WatchGrafanaStacksChange(tick.C); err != nil { + setupLog.Error(err, "unable to create grafana stacks reconciler", "controller", "GrafanaStacks") + os.Exit(1) + } + // ############################################# + + // ############## Set up Namespace Reconciler ############## + if err = (&controllers.NamespaceReconciler{ + Client: mgr.GetClient(), + Log: ctrl.Log.WithName("controllers").WithName("Namespace"), + TracesNamespace: tracesNamespace, + ExcludeWorkloadLabelSelector: appSelector, + ExcludeNamespaceLabelSelector: namespaceSelector, + IgnoreApps: strings.Split(ignoreApps, ","), + IgnoreNamespaces: strings.Split(ignoreNamespaces, ","), + GrafanaCloudUpdater: gcConfigUpdater, + GrafanaCloudClient: gcClient, + GrafanaCloudTracesToken: grafanaCloudTracesToken, + ClusterName: clusterName, + EnableVPA: enableVpa, + }).SetupWithManager(mgr, grafanaStackChangeEvents); err != nil { + setupLog.Error(err, "unable to create controller", "controller", "Namespace") + os.Exit(1) + } + // ############################################# + + // ############## Set up PodMonitor Reconciler ############## + if err = (&controllers.PodMonitorReconciler{ + Client: mgr.GetClient(), + Log: ctrl.Log.WithName("controllers").WithName("PodMonitor"), + Scheme: mgr.GetScheme(), + ClusterName: clusterName, + Region: clusterRegion, + GrafanaCloudCredentials: grafanaCloudCredentials, + GrafanaCloudClient: gcClient, + PrometheusNamespace: promNamespace, + PrometheusExposedDomain: fmt.Sprintf("%s.%s", clusterName, clusterDomain), + NodeSelectorTarget: prometheusNodeSelectorTarget, + EnableMetricsRemoteWrite: enableMetricsRemoteWrite, + EnableVpa: enableVpa, + PrometheusDockerImage: controllers.DockerImage{ + Name: prometheusDockerImage, + Tag: prometheusDockerTag, + }, + PrometheusServiceAccountName: prometheusServiceAccountName, + PrometheusPodPriorityClassName: prometheusPodPriorityClassName, + PrometheusMonitoringTarget: prometheusMonitoringTargetName, + PrometheusExtraExternalLabels: prometheusExtraExternalLabelsMap, + }).SetupWithManager(mgr); err != nil { + setupLog.Error(err, "unable to create controller", "controller", "PodMonitor") + os.Exit(1) + } + // ############################################# + + // +kubebuilder:scaffold:builder + + setupLog.Info("starting manager") + if err := mgr.Start(ctrl.SetupSignalHandler()); err != nil { + setupLog.Error(err, "problem running manager") + os.Exit(1) + } +} diff --git a/cmd/observability-operator/main_test.go b/cmd/observability-operator/main_test.go new file mode 100644 index 0000000..d3bcd14 --- /dev/null +++ b/cmd/observability-operator/main_test.go @@ -0,0 +1,164 @@ +package main + +import ( + "context" + "fmt" + "os" + "os/exec" + "strings" + "testing" + "time" + + k8s "github.com/adevinta/go-k8s-toolkit" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/e2e-framework/pkg/env" + "sigs.k8s.io/e2e-framework/pkg/envconf" + "sigs.k8s.io/e2e-framework/pkg/envfuncs" + "sigs.k8s.io/e2e-framework/support/kind" + "sigs.k8s.io/e2e-framework/third_party/helm" +) + +var ( + testenv env.Environment + imageRegistry = "local" + imageRepository = "adevinta/observability-operator" + imageTag = "latest" + imageFullUrl = imageRegistry + "/" + imageRepository + ":" + imageTag + kindClusterName = envconf.RandomName("observability", 16) + releaseName = "observability-operator" + + operatorNamespace = "observability-operator" + prometheusNamespace = "platform-services" + tracesNamespace = "observability" +) + +func buildLocalObservabilityOperatorImage(t *testing.T) { + // Ensure we always have the latest version of the code compiled + // This is crucial when running integration tests locally + cmd := exec.Command("docker", "build", "-t", imageFullUrl, ".") + cmd.Stderr = os.Stderr + cmd.Stdout = os.Stdout + cmd.Dir = "../../" + require.NoError(t, cmd.Run()) + + cmd = exec.Command("kind", "load", "docker-image", "--name", kindClusterName, imageFullUrl) + cmd.Stderr = os.Stderr + cmd.Stdout = os.Stdout + cmd.Dir = "../../" + require.NoError(t, cmd.Run()) +} + +func installObservabilityOperator(t *testing.T, args ...string) { + t.Helper() + helmClient := helm.New(testenv.EnvConf().KubeconfigFile()) + + require.NoError(t, helmClient.RunUpgrade( + helm.WithName(releaseName), + helm.WithNamespace(operatorNamespace), + helm.WithChart("../../helm-chart/observability-operator"), + helm.WithArgs( + "--install", + ), + helm.WithArgs(args...), + )) +} + +// Verify the Helm chart can complete successfully, the pod can be +// scheduled and get to ready. The Pod may fail right afterwards due +// to missing credentials or values, but it was able to start-up, so +// the CLI flags, Helm values and immediate verifications on startup +// passed. +func TestDeployDefaultHelmChart(t *testing.T) { + t.Setenv("KUBECONFIG", testenv.EnvConf().KubeconfigFile()) + + secretName := "observability-operator-grafana-cloud-credentials" + + buildLocalObservabilityOperatorImage(t) + installObservabilityOperator( + t, + "--set", "image.registry="+imageRegistry, + "--set", "image.repository="+imageRepository, + "--set", "image.tag="+imageTag, + "--set", "image.pullPolicy=Never", + "--set", "enableSelfVpa=false", + "--set", "namespaces.tracesNamespace.name="+tracesNamespace, + "--set", "namespaces.prometheusNamespace.name="+prometheusNamespace, + "--set", "credentials.GRAFANA_CLOUD_TOKEN.secretName="+secretName, + "--set", "credentials.GRAFANA_CLOUD_TRACES_TOKEN.secretName="+secretName, + ) + + cfg, err := k8s.NewClientConfigBuilder().WithKubeConfigPath(testenv.EnvConf().KubeconfigFile()).Build() + require.NoError(t, err) + k8sClient, err := client.New(cfg, client.Options{}) + require.NoError(t, err) + + secret := corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: secretName, + Namespace: operatorNamespace, + }, + StringData: map[string]string{ + "grafana-cloud-api-key": "", + "grafana-cloud-traces-token": "", + }, + } + err = k8sClient.Create(context.Background(), &secret) + require.NoError(t, err) + + assert.Eventually( + t, + func() bool { + podList := corev1.PodList{} + err = client.NewNamespacedClient(k8sClient, operatorNamespace).List(context.Background(), &podList) + require.NoError(t, err) + + require.Len(t, podList.Items, 1) + return strings.HasPrefix(podList.Items[0].Name, releaseName) + }, + 30*time.Second, 10*time.Millisecond, + "The operator pod should be present", + ) + + assert.Eventually( + t, + func() bool { + podList := corev1.PodList{} + err = client.NewNamespacedClient(k8sClient, operatorNamespace).List(context.Background(), &podList) + require.NoError(t, err) + + require.Len(t, podList.Items, 1) + return corev1.PodPhase("Running") == podList.Items[0].Status.Phase + }, + 30*time.Second, 10*time.Millisecond, + "The operator pod should be running", + ) +} + +func TestMain(m *testing.M) { + if os.Getenv("RUN_INTEGRATION_TESTS") != "true" { + fmt.Printf("RUN_INTEGRATION_TESTS is not set, so skipping all tests of main") + os.Exit(0) + } + + testenv = env.New() + // Use pre-defined environment funcs to create a kind cluster prior to test run + testenv.Setup( + envfuncs.CreateCluster(kind.NewCluster(kindClusterName), kindClusterName), + envfuncs.CreateNamespace(operatorNamespace), + envfuncs.CreateNamespace(prometheusNamespace), + envfuncs.CreateNamespace(tracesNamespace), + ) + + // Use pre-defined environment funcs to teardown kind cluster after tests + testenv.Finish( + envfuncs.DeleteNamespace(operatorNamespace), + envfuncs.DestroyCluster(kindClusterName), + ) + + // launch package tests + os.Exit(testenv.Run(m)) +} diff --git a/docs/01-send-pod-metrics/README.md b/docs/01-send-pod-metrics/README.md new file mode 100644 index 0000000..aeb2f31 --- /dev/null +++ b/docs/01-send-pod-metrics/README.md @@ -0,0 +1,84 @@ +# Send Pod metrics to Grafana Cloud + +> [!NOTE] +> Administrators can modify the domain of the cluster, which will affect the domain used by label and annotation keys. +> If the cluster's domain is `example.com` the labels and annotations will match it, e.g. `grafanacloud.example.com/stack-name`. +> +> In this document we'll use `adevinta.com` as the cluster domain. + +## Overview + +To send [`Pod`][k8s-pod] metrics to Grafana Cloud, you'll need two steps: + +1. To configure your team's Grafana Cloud stack (account) in your [`Namespace`][k8s-namespace] definition, using the **[annotation][k8s-annotation]** `grafanacloud.adevinta.com/stack-name: `. +2. To annotate your [`Pods`][k8s-pod] with the **[annotations][k8s-annotation]**: + 1. `prometheus.io/scrape: "true"` (**Mandatory**, _string_). This will tell us that we must ingest the [`Pod`][k8s-pod]'s metrics + 2. `prometheus.io/path: ` (**Mandatory**, _string_). The path where Prometheus should get the metrics. + 3. `prometheus.io/port: "9090"` (**Optional**, _string_). The port where Prometheus should get the metrics. It may be the same as your pod, if you have an HTTP service, or it may be a different one. By default, Prometheus will scrape, by default, all exposed container ports. In most cases, you don't need this configuration. + 4. `monitor.adevinta.com/pod-sample-limit: ` (**Optional**, _string_). This will tell the sample limit allowed by the prometheus for that pod, by default is set to 4500. (this flag is still experimental and could change in the future) + +![Send metrics diagram](../images/send-metrics-grafana-cloud.png) + +## Configuring your namespace + +Below there's an example of how to configure your [`Namespace`][k8s-namespace]: + +```diff +apiVersion: v1 +kind: Namespace +metadata: + # Ommitted for brevity + annotations: ++ grafanacloud.adevinta.com/stack-name: mymicroservicestack +``` + +> [!NOTE] +> By leveraging the `grafanacloud.adevinta.com/stack-name` annotation, your team can have multiple namespaces using the same Grafana Cloud stack. + +After the configuration is applied, the metrics will start flowing to your Grafana Cloud stack. + +## Using multiple Grafana stacks + +Below there's an example of how to configure your [`Namespace`][k8s-namespace] to forward metrics to multiple Grafana stacks: + +```diff +apiVersion: v1 +kind: Namespace +metadata: + # Ommitted for brevity + annotations: ++ grafanacloud.adevinta.com/stack-name: firststack,secondstack +``` + +> [!NOTE] +> By leveraging the `grafanacloud.adevinta.com/stack-name` annotation, your team can have multiple namespaces using **multiple** Grafana Cloud stacks. + +After the configuration is applied, the metrics will start flowing to your Grafana Cloud stacks. + +## Disable metrics or logs to be sent to your Grafana Cloud Stack + +You have the option to disable logs, metrics, or both from being sent to your Grafana Cloud Stack, giving you complete flexibility in choosing what to utilize. + +### Disable metrics: +```diff +apiVersion: v1 +kind: Namespace +metadata: + # Ommitted for brevity + annotations: ++ grafanacloud.adevinta.com/metrics: disabled +``` + +### Disable logs: +```diff +apiVersion: v1 +kind: Namespace +metadata: + # Ommitted for brevity + annotations: ++ grafanacloud.adevinta.com/logs: disabled +``` + +[k8s-annotation]: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ +[k8s-pod]: https://kubernetes.io/docs/concepts/workloads/pods/ +[k8s-namespace]: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/ diff --git a/docs/02-send-pod-metrics-different-backend/README.md b/docs/02-send-pod-metrics-different-backend/README.md new file mode 100644 index 0000000..9c8a503 --- /dev/null +++ b/docs/02-send-pod-metrics-different-backend/README.md @@ -0,0 +1,59 @@ +# Send Pod metrics to a different backend + +## Overview + +By default, the operator sends metrics to the corresponding Grafana Cloud stack, either [via convention or configuration](../01-send-pod-metrics/README.md). + +You can change this behaviour, setting a custom [storage][prometheus-storage-docs] (`remoteWrite`). For instance, this may be useful to configure a [Victoria Metrics][victoria-metrics] backend to store your metrics: + +![](../images/send-data-different-backend.png) + +[Zoom](../images/send-data-different-backend.png) | [Diagram source](../images/source/send-data-different-backend.excalidraw) | [Shareable link][send-metrics-different-backend-shareable-link] + +## Configuring the backend + +If you want to use this feature, you need to annotate your [`Namespace`][k8s-namespace], with a [`remoteWrite`][prometheus-remote-write-docs] configuration: + +```diff +apiVersion: v1 +kind: Namespace +metadata: + annotations: + # Ommitted for brevity ++ metrics.monitoring.adevinta.com/remote-write={namespace}/{secret-name} +``` + +In the above example, `{secret-name}` is a reference to a [Secret][k8s-secret] created in the cluster. + +> [!NOTE] +> If the [Secret][k8s-secret] is defined in the same namespace where it's being configured, then you do not need to namespace it. So, in that case, you may eliminate the `{namespace}/` part. + +The content of the secret is a [`remoteWrite`][prometheus-remote-write-docs] spec as well as references to secrets that need to be used: + +```yaml +apiVersion: v1 +kind: Secret +metadata: + name: my-remote-write-secret + annotations: + monitor.adevinta.com/referenced-secrets: my-secret,my-secret-2 +stringData: + remote-write: | + url: https://my-remote-write + basicAuth: + username: + name: my-secret + key: user-name + password: + name: my-secret-2 + key: password +``` + +Observability Operator then configures the tenant's Prometheus [`remoteWrite`][prometheus-remote-write-docs] accordingly. + +[k8s-namespace]: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/ +[k8s-secret]: https://kubernetes.io/docs/concepts/configuration/secret/ +[prometheus-storage-docs]: https://prometheus.io/docs/prometheus/latest/storage/#remote-storage-integrations +[prometheus-remote-write-docs]: https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write +[send-metrics-different-backend-shareable-link]: https://excalidraw.com/#json=Jj-IkULKwAafn4b_IHygp,YzbiPd5xoJBGRfbuddInOA +[victoria-metrics]: https://victoriametrics.com/ \ No newline at end of file diff --git a/docs/03-configure-loki/README.md b/docs/03-configure-loki/README.md new file mode 100644 index 0000000..9b09713 --- /dev/null +++ b/docs/03-configure-loki/README.md @@ -0,0 +1,70 @@ +# Configure the logging backend for Grafana Cloud's Loki + +> [!NOTE] +> Administrators can modify the domain of the cluster, which will affect the domain used by label and annotation keys. +> If the cluster's domain is `example.com` the labels and annotations will match it, e.g. `grafanacloud.example.com/stack-name`. +> +> In this document we'll use `adevinta.com` as the cluster domain. + +## Overview + +To send [`Pod`][k8s-pod] logs to Grafana Cloud, you'll need one step: + +1. To configure your team's Grafana Cloud stack (account) in your [`Namespace`][k8s-namespace] definition, using the **[annotation][k8s-annotation]** `grafanacloud.adevinta.com/stack-name: `. + +![Log architecture diagram](../images/logs-overview.png) + +## Configuring your namespace + +Below there's an example of how to configure your [`Namespace`][k8s-namespace]: + +```diff +apiVersion: v1 +kind: Namespace +metadata: + # Ommitted for brevity + annotations: ++ grafanacloud.adevinta.com/stack-name: mystack +``` + +> [!NOTE] +> By leveraging the `grafanacloud.adevinta.com/stack-name` annotation, your team can have multiple namespaces using the same Grafana Cloud stack. + +After the configuration is applied, the logs will start flowing to your Grafana Cloud stack. + +## Using multiple Grafana stacks + +Below there's an example of how to configure your [`Namespace`][k8s-namespace] to forward logs to multiple Grafana Cloud stacks: + +```diff +apiVersion: v1 +kind: Namespace +metadata: + # Ommitted for brevity + annotations: ++ grafanacloud.adevinta.com/stack-name: firststack,secondstack +``` + +> [!NOTE] +> By leveraging the `grafanacloud.adevinta.com/stack-name` annotation, your team can have multiple namespaces using **multiple** Grafana Cloud stacks. + +After the configuration is applied, the logs will start flowing to your Grafana Cloud stacks. + +## Opting out from logs + +If you want to opt out from having logs in your Loki, you'll need to annotate your [`Pods`][k8s-pod]: + +```diff +apiVersion: apps/v1 +kind: Deployment +spec: + template: + spec: + metadata: + annotations: ++ grafanacloud.adevinta.com/logs: disabled +``` + +[k8s-annotation]: https://kubernetes.io/docs/concepts/overview/working-with-objects/annotations/ +[k8s-pod]: https://kubernetes.io/docs/concepts/workloads/pods/ +[k8s-namespace]: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/ \ No newline at end of file diff --git a/docs/05-send-pod-traces/README.md b/docs/05-send-pod-traces/README.md new file mode 100644 index 0000000..521ecb4 --- /dev/null +++ b/docs/05-send-pod-traces/README.md @@ -0,0 +1,131 @@ +# Send Pod traces to Grafana Cloud + +> [!NOTE] +> Administrators can modify the domain of the cluster, which will affect the domain used by label and annotation keys. +> If the cluster's domain is `example.com` the labels and annotations will match it, e.g. `grafanacloud.example.com/stack-name`. +> +> In this document we'll use `adevinta.com` as the cluster domain. + +## Overview + +To start using traces, users need to: + +1) setup the namespaces where traces are to be enabled + +2) instrument the application(s) to send traces to the corresponding OTEL collector. + +> [!NOTE] +> Currently, the tracing integration only supports **Grafana Cloud Tempo** as backend for traces. + +## Configuring your namespaces + +Your namespaces must have the `grafanacloud.adevinta.com/stack-name` annotation which specifies the Grafana Cloud +stack that your namespaces will use for sending metrics, logs and traces. + +In addition to the `grafanacloud.adevinta.com/stack-name`, add the `grafanacloud.adevinta.com/traces` annotation to enable tracing: + +```yaml +metadata: + annotations: + grafanacloud.adevinta.com/stack-name: "{my-grafana-cloud-stack-name}" + grafanacloud.adevinta.com/traces: "enabled" +``` + +This will setup a dedicated OTel Collector for each configured namespace. + +Collectors will be accessible from the user specified namespaces via the following endpoint: + +* `otelcol-{namespace}.observability.svc.cluster.local` + +and through ports: + +* `4317` (for OTLP/gRPC connections) +* `4318` (for OTLP/HTTP connections) + +## Instrumenting your apps + +Once you have your [namespace ready](#Configuring-your-namespaces), you need to setup intrumentation in your apps to send traces to +your OTel Collector(s). + +Users may configure the OTLP endpoint and protocol either through the OTel SDK and languange of their choice[^3] or via +environment variables[^4]. + +An example follows using environment variables: + +* using OTLP/gRPC + + ```yaml + # Deployment manifest + spec: + containers: + - name: my-instrumented-app + image: hello-app:1.0 + env: + - name: OTEL_EXPORTER_OTLP_PROTOCOL + value: "grpc" + - name: OTEL_EXPORTER_OTLP_ENDPOINT + value: "http://otelcol-{namespace}.observability.svc.cluster.local:4317" + ``` + +* using OTLP/HTTP + + ```yaml + # Deployment manifest + spec: + containers: + - name: my-instrumented-app + image: hello-app:1.0 + env: + - name: OTEL_EXPORTER_OTLP_PROTOCOL + value: "http/protobuf" + - name: OTEL_EXPORTER_OTLP_ENDPOINT + value: "http://otelcol-{namespace}.observability.svc.cluster.local:4318" + ``` + +> [!WARNING] +> If you set the OTLP endpoint in your application code through the Otel SDK (instead of using environment variables) you may +> need to append the `/v1/traces` path. For example in Python[^5] this might look like this: +> +> ```python +> from opentelemetry.exporter.otlp.proto.http.trace_exporter import OTLPSpanExporter +> +> otlp_http_exporter = OTLPSpanExporter( +> endpoint="http://otelcol-{namespace}.observability.svc.cluster.local:4318/v1/traces" +> ) +> ``` +> +> Check the corresponding SDK documentation[^3] for more details. + +## Disabling traces + +You can delete your dedicated OTel Collector at any time by annotating your namespaces as follows: + +```yaml +metadata: + annotations: + grafanacloud.adevinta.com//traces: "disabled" +``` + +Or simply removing the whole `grafanacloud.adevinta.com/traces` annotation. + +Before deleting your OTel Collector make sure your app instrumenation is no longer sending traces to the collector, otherwise your +app will error out when trying to send spans. + +## Sampling + +The Operator provides the following default sampling strategy to all managed OTel Collectors: + +* successful traces (`status_codes = ["OK", "UNSET"]`): + only 10% of these traces are sampled + +* traces containing errors (`status_codes = ["ERROR"]`): + 100% of these traces are sampled + +This means that for a service generating 100 succesfull traces/sec only 10 traces/sec will be stored +in the backend together with any other traces that contain errors. + +[^1]: +[^2]: +[^3]: +[^4]: +[^5]: diff --git a/docs/README.md b/docs/README.md new file mode 100644 index 0000000..0d19f52 --- /dev/null +++ b/docs/README.md @@ -0,0 +1,32 @@ +# Observability Operator + +Observability Operator allows to dynamically create [`PodMonitor`][prom-operator-podmonitor-docs] and [`Prometheus`][prom-operator-prometheus-docs] +objects, based on Prometheus Operator, in a Kubernetes cluster. + +These Prometheuses can then use a `remoteWrite` to send the metrics to a backend such as [Grafana Cloud][grafana-cloud] or +[Victoria Metrics][victoria-metrics]. + +Observability Operator can also dynamically create OTel Collectors for those user namespaces that request tracing capabilities. + +If you're looking into how to contribute to this project, look into the [contributing guidelines](../CONTRIBUTING.md). + +## Overview + +Below there's an image with an architecture overview of the metrics, traces and logging collection using Observability Operator: + +[![architecture overview](images/architecture-overview.png)](images/architecture-overview.png) + +## Features + +There are four main features provided by the Observability Operator: + +1. [Send Pod metrics to Grafana Cloud](01-send-pod-metrics/README.md) +2. [Send Pod metrics to a different backend](02-send-pod-metrics-different-backend/README.md), such as [Victoria Metrics][victoria-metrics] +3. [Configure the logging backend for Grafana Cloud's Loki](03-configure-loki/README.md) +4. [Manage synthetic monitors for your Ingresses](04-manage-synthetic-monitors-ingress/README.md) +5. [Send Pod traces to Grafana Cloud](05-send-pod-traces/README.md) + +[grafana-cloud]: https://grafana.com/products/cloud/ +[prom-operator-podmonitor-docs]: https://prometheus-operator.dev/docs/operator/design/#podmonitor +[prom-operator-prometheus-docs]: https://prometheus-operator.dev/docs/operator/design/#prometheus +[victoria-metrics]: https://victoriametrics.com/ \ No newline at end of file diff --git a/docs/administrators/README.md b/docs/administrators/README.md new file mode 100644 index 0000000..63e148f --- /dev/null +++ b/docs/administrators/README.md @@ -0,0 +1,200 @@ +# Observability Operator for administrators + +This documentation explains in detail how to operate Grafana Cloud +Operator in your cluster(s). + +## Cluster Requirements + +### Runtime Dependencies + +Observability Operator can collect metrics, traces and logs. It +manages the entire requirements to gather traces, but has dependencies +for the other cases. + +| Dependency | Operator version | Dependency version | +| ----------------------- | ------------------------------------- | ---------------------------------- | +| `Prometheus Operator` | > `v0.31.0` (verified with `v0.74.0`) | verified with Prometheus `v2.43.0` | +| `Kube Fluentd Operator` | verified with `v1.18.1` | verified with fluentd `1.16.1` | + +#### `Prometheus Operator` + +Observability Operator relies on the `monitoring.coreos.com/v1` resources +provided by `Prometheus Operator` CRDs. These resources were originally introduced +in version `v0.31.0` of the `Prometheus Operator`. + +Observability Operator currently manages (through `Prometheus Operator`) Prometheus instances based +on version `v2.43.0`. This version of Prometheus might be customized in Observability Operator's helm values +via the `prometheusVersion` variable. + +#### `Kube Fluentd Operator` + +Observability Operator relies on `fluentd` to forward tenant logs to their corresponding storage backend. Tenants must annotate their namespace to specify which Grafana Cloud stack their logs should be sent to. + +To enable this integration with Grafana Cloud, the Observability Operator dynamically updates a Configmap (named `grafana-cloud-fluentd-config`) with the required fluentd rules[^1]. + +`Kube Fluentd Operator` must load these rules from the `grafana-cloud-fluentd-config` configmap into `fluentd`s configuration, otherwise logs wont reach any of tenant stacks in Grafana Cloud. + +This requires `Kube Fluentd Operator` configuration to follow these steps: + +* define a volume in the `fluentd` pod that loads the Observability Operator's configmap and loki key/paths: + +```yaml + - name: loki-config + configMap: + name: grafana-cloud-fluentd-config + optional: true + items: + - key: loki + path: loki +``` + +* mount the contents of the previously defined volume as a `/templates/loki.conf` file in the `fluentd` pod: + +```yaml + - name: loki-config + mountPath: /templates/loki.conf + subPath: loki +``` + +* add the following directive into `fluentd`'s `fluent.conf` main configuration to load Observability Operator managed rules, stored in `grafana-cloud-fluentd-config` configmap: + +```xml + @include loki.conf +``` + +In addition to this, log events that shall be forwarded to Grafana Cloud must be tagged with a `loki.` prefix. + +The following `fluentd` directives show how to achieve this: + +> [!NOTE] +> This setup assumes logs are collected at the node level by `fluentbit` agents and forwarded to a `fluentd` service managed by `Kube Fluentd Operator`): + +```xml + + @type record_modifier + remove_keys _dummy_ + + kubernetes_namespace_container_name ${record["kubernetes"]["namespace_name"]}.${record["kubernetes"]["pod_name"].split('.')[-1]}.${record["kubernetes"]["container_name"]} + # dark magic to change nested attributes, this record is removed inmediately, it is only used to change kubernetes.pod_name value + _dummy_ ${record['kubernetes']['pod_name'] = record['kubernetes']['pod_name'].split('.')[-1]; nil} + + + + # retag based on the namespace and container name of the log message + + @type rewrite_tag_filter + @id record_tag_rewrite + # Update the tag have a structure of kube.. + + key kubernetes_namespace_container_name + pattern ^(.+)$ + tag kube.$1 + + + + + @type rewrite_tag_filter + @id copy_log_to_loki + + key $.kubernetes.container_name + pattern /.+/ + tag loki.${tag} + + +``` + +The previous snippet will: + +1. tag container logs (captured via fluentbit node agents) with `kube.` tag +2. prepend a `loki.` tag to the container logs that originated from namespaces with log collection enabled. + +Users who are not using Grafana Cloud can send their logs to a different storage backend by creating the corresponding fluentd rules in a Configmap in their namespace. This Configmap must be named `fluentd-config`. + +> [!WARNING] +> By default, `Kube Fluentd Operator` compiles all ConfigMaps named `fluentd-config` from every namespace to create the final Fluentd configuration. Any syntax error in these ConfigMaps will affect the entire `Fluentd` log processing pipeline. + +A simplified example for sending logs to a custom fluentd backend follows: + +```yaml +apiVersion: v1 +kind: ConfigMap +metadata: + name: fluentd-config +data: + fluent.conf: | + + @type forward + @id _match_forward + + name forwarder-server + host + port + + … + +``` + +### Networking + +You have to provide some namespaces for the operator to allocate the +Prometheus storage systems and the OpenTelemetry collectors. We +recommend to use separate namespaces for the Prometheus and the +OpenTelemetry collectors as they have different networking access +requirements (pull-model vs push-model). + +#### Networking requirements for Metrics + +The namespace for the Prometheus systems should be allowed to access +all tenant namespaces, so it can scrape the workloads. This is +expected to be provisioned by the cluster owners. + +#### Networking requirements for Traces + +For traces, the operator provisions NetworkPolicy objects to grant the +required accesses between the namespaces holding the workloads and the +namespaces holding the collectors and Prometheus deployments. + +This scheme assumes: + +- your cluster has the components required to enforce these + NetworkPolicy objects +- communication between namespace objects is blocked by default + (allow-list pattern) + +The NetworkPolicy objects created restrict access to the +OpentTelemetry collector's Service for a specific namespace to +workloads of that namespace, matching collectors and namespaces 1:1. + +## Helm Chart + +The Helm chart provides the following options: + +| Option | Required? | Defaults | Description | +| ----------------------------------- | --------- | ---------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| replicaCount | Yes | 1 | How many operator replicas to deploy | +| image.registry | Yes | | The registry hosting the operator image. | +| image.repository | Yes | | The path to the operator image inside that registry. | +| image.tag | Yes | | The tag of the operator image to be deployed. | +| image.imagePullPolicy | Yes | | The operator image pull policty used by the Operator pod. | +| region | No | eu-west-1 | Metadata added to telemetry data | +| clusterName | No | CHANGEME | Metadata added to telemetry data | +| clusterDomain | Yes | CHANGEME | The cluster domain. Used to generate the name of labels and annotations used to configure the operator behaviour as well as internal consistency. | +| roleARN | Yes | CHANGEME | The AWS ARN for the IAM role the Operator should adopt. This is required to grant interaction with the AWS API. | +| exclusionLabelSelectors.workload | No | | List of workload label selectors that should be ignored. Deployments of this name will be skipped despite how they are configured. | +| exclusionLabelSelectors.namespace | No | | List of namespace label selectors that should be ignored. The telemetry of workloads present in the listed namespaces will be ignored no matter how they are configured. | +| enableGrafanaCloud | Yes | false | Whether to set up the discovered GrafanaCloud stacks as destination for telemetry. If disabled, fallbacks have to be setup on each individual namespace for each telemetry type. | +| enableVPA | No | false | Whether to allocate VPA objects for the Prometheus and Alloy deployments that collect tenant telemetry. | +| prometheusDockerImage.tag | Yes | v2.43.0 | Tag of the Prometheus image to be used when provisioning Prometheus statefulsets for tenant metrics scraping and storage. | +| service.internalPort | Yes | 8080 | Internal port for exposing Operator metrics in prometheus format | +| enableSelfVpa | No | true | Create a VPA object for the operator itself | +| resources.limits.cpu | No | 100m | The CPU limits of the operator deployment | +| resources.limits.memory | No | 128Mi | The memory limits of the operator deployment | +| resources.requests.cpu | No | 100m | The CPU requests of the operator deployment | +| resources.requests.memory | No | 128Mi | The memory requests of the operator deployment | +| grafanacloud.configmap.namespace | Yes | | The namespace holding the configmap with Loki credentials to send logs to GrafanaCloud. The credentials are the same for all tenants in the cluster. | +| grafanacloud.configmap.name | Yes | | The name of the configmap with Loki credentials to send logs to GrafanaCloud. The credentials are the same for all tenants in the cluster. | +| grafanacloud.configmap.lokikey | Yes | | The configmap key holding Loki credentials to send logs to GrafanaCloud. The credentials are the same for all tenants in the cluster. | +| namespaces.tracesNamespace.name | Yes | observability | The name of the provisioned namespace where Alloy deployments will be allocated to | +| namespaces.prometheusNamespace.name | Yes | platform-services | The name of the provisioned namespace where Prometheus deployments will be allocated to | + +[^1]: diff --git a/docs/development/README.md b/docs/development/README.md new file mode 100644 index 0000000..743957c --- /dev/null +++ b/docs/development/README.md @@ -0,0 +1,226 @@ +# Developing observability-operator + +Make sure you have Golang already setup in your local environment. + +## Running tests + +We have 4 different kinds of tests: +- Standard unit tests at reconciler level +- Linter/formatting tests for embedded configuration files +- Integration tests with GrafanaCloud API + - Require a valid API key set as env variable GRAFANA_CLOUD_TOKEN + - Verify the Grafana.com API client works properly (for stack discovery) +- Integration tests for the entire program and Helm chart run in a Kind cluster + - Require env variable RUN_INTEGRATION_TESTS set to "true" + - Verify the operator can be deployed and behave correctly inside a k8s cluster + +## Setup a local Kubernetes cluster with Kind + +For convenience, you may want to run the Operator against a local cluster during development. + +Make sure you have a container runtime installed. + +For macOS users, consider using `colima`[^1]: + +```bash +colima start --edit # set enough resources, ie: 4 cpus and 8gb mem +``` + +### Create kind cluster + +* Create a Kind cluster configuration (ie: `kind-cluster.yaml`): + + ```bash + cat << EOF > kind-cluster.yaml + # three node (two workers) cluster config + kind: Cluster + apiVersion: kind.x-k8s.io/v1alpha4 + name: observability-operator-demo + nodes: + - role: control-plane + image: kindest/node:v1.30.0 + - role: worker + image: kindest/node:v1.30.0 + EOF + ``` + +* Create the Kind cluster: + + ```bash + kind create cluster --config kind-cluster.yaml + ``` + +### Setup Operator runtime dependencies + +> [!IMPORTANT] +> Make sure you are using the kubectl context of your kind cluster, ie: +> +> ```bash +> kubectl config use-context kind-observability-operator-demo # or equivalent with kubectx +> ``` + +* Namespaces + + ```bash + kubectl create namespace platform-services # for Prometheus instances and operator + kubectl create namespace observability # for Alloy instances + kubectl create namespace userns-dev # for user namespace (example) + kubectl create namespace observability-operator # where Operator pods run (if installed via helm) + ``` + +* Prometheus Operator + + ```bash + helm repo add prometheus-community https://prometheus-community.github.io/helm-charts + helm install prometheus-operator prometheus-community/kube-prometheus-stack -n platform-services + ``` + +* Kubernetes Metrics Server + + ```bash + curl -L -O https://github.com/kubernetes-sigs/metrics-server/releases/download/v0.7.2/components.yaml + + # edit components.yaml: + # add `- --kubelet-insecure-tls` arg to "metrics-server" container + + kubectl apply -f components.yaml + ``` + + You should now be able to run `kubectl top pod` against your kind cluster. + +* Vertical Pod Autoscaler (VPA) + + ```bash + git clone https://github.com/kubernetes/autoscaler.git + + # fix addext param for openssl version https://github.com/kubernetes/autoscaler/issues/6126if + OPENSSL_VERSION=$(openssl version | awk '{print $2}') + if [[ ! $OPENSSL_VERSION =~ ^1\.1\..* ]]; then + echo "OpenSSL version is not 1.1.x, modifying the file..." + sed -i 's/-addext "subjectAltName = DNS:${CN_BASE}_ca"//g' autoscaler/vertical-pod-autoscaler/pkg/admission-controller/gencerts.sh + else + echo "OpenSSL version is 1.1.x, no modification needed." + fi + + # Make the autoscaler up + ./autoscaler/vertical-pod-autoscaler/hack/vpa-up.sh + # check vpa-admission-controller is running + kubectl get pods -n kube-system + ``` + +## Run the Operator + +> [!IMPORTANT] +> Make sure you are using the kubectl context of your kind cluster, ie: +> +> ```bash +> kubectl config use-context kind-observability-operator-demo # or equivalent with kubectx +> ``` + +### A. Locally against your local kind cluster + +Tokens must be set but they can be fake if you don't need to test the Grafana Cloud integration. + +```bash +GRAFANA_CLOUD_TOKEN='test1234' GRAFANA_CLOUD_TRACES_TOKEN='123455'\ +go run cmd/observability-operator/main.go\ + -cluster-name='observability-operator-demo'\ + -cluster-region='eu-west-1'\ + -cluster-environment='dev'\ + -cluster-domain='adevinta.com' +``` + +### B. In-cluster (within your local kind cluster) + +```bash +docker build -t local/adevinta/observability-operator:latest . +kind load docker-image --name observability-operator-demo local/adevinta/observability-operator:latest + +cat << EOF > secrets.yaml +kind: Secret +type: Opaque +apiVersion: v1 +metadata: + name: observability-operator-grafana-cloud-credentials + namespace: observability-operator +data: + grafana-cloud-api-key: YOUR_BASE64_ENCODED_TOKEN + grafana-cloud-traces-token: YOUR_BASE64_ENCODED_TOKEN +EOF + +kubectl apply -f secrets.yaml + +helm upgrade observability-operator helm-chart/observability-operator --install --namespace observability-operator\ + --set image.registry=local\ + --set image.repository=adevinta/observability-operator\ + --set image.tag=latest\ + --set image.pullPolicy=Never\ + --set enableSelfVpa=false\ + --set clusterName=observability-operator-demo\ + --set region=eu-west-1\ + --set clusterDomain=adevinta.com\ + --set excludeNamespaces=kube-system\ + --set credentials.GRAFANA_CLOUD_TOKEN.secretName=observability-operator-grafana-cloud-credentials\ + --set credentials.GRAFANA_CLOUD_TRACES_TOKEN.secretName=observability-operator-grafana-cloud-credentials +``` + +## [Optional] Debug the Operator with VS Code + +If you are using VS Code you can use the following "launch configuration"[^2] to start a Debugging session from +within your editor: + +```json +// .vscode/launch.json +{ + "version": "0.2.0", + "configurations": [ + { + "name": "Run Observability Operator", + "type": "go", + "request": "launch", + "mode": "auto", + "program": "${workspaceFolder}/cmd/observability-operator/main.go", + "args": [ + "-cluster-name", "observability-operator-demo", + "-cluster-region", "eu-west-1", + "-cluster-environment", "dev", + ], + "env": { + "GRAFANA_CLOUD_TOKEN": "test1234", + "GRAFANA_CLOUD_TRACES_TOKEN": "test1234" + }, + "console": "integratedTerminal", + "preLaunchTask": "check-kubectl-context" + } + ] +} +``` + +The `check-kubectl-context` preLaunchTask might be useful to ensure you are running against the right kubernetes cluster. + +You may define this task as follows: + +```json +// your workspace settings +{ + "folders": [ + { + "path": "path/to/observability-operator" + } + ], + "settings": {}, + "tasks": { + "version": "2.0.0", + "tasks": [ + { + "label": "check-kubectl-context", + "type": "shell", + "command": "[[ $(kubectl config current-context) == 'kind-observability-operator-demo' ]] || exit 1" + } + ] + } +} +``` + +[^1]: +[^2]: diff --git a/docs/images/architecture-overview.png b/docs/images/architecture-overview.png new file mode 100644 index 0000000..361ddff Binary files /dev/null and b/docs/images/architecture-overview.png differ diff --git a/docs/images/logs-overview.png b/docs/images/logs-overview.png new file mode 100644 index 0000000..6bc3bb3 Binary files /dev/null and b/docs/images/logs-overview.png differ diff --git a/docs/images/send-data-different-backend.png b/docs/images/send-data-different-backend.png new file mode 100644 index 0000000..ae9b81c Binary files /dev/null and b/docs/images/send-data-different-backend.png differ diff --git a/docs/images/send-metrics-grafana-cloud.png b/docs/images/send-metrics-grafana-cloud.png new file mode 100644 index 0000000..b7c01b3 Binary files /dev/null and b/docs/images/send-metrics-grafana-cloud.png differ diff --git a/docs/images/source/architecture-overview.excalidraw b/docs/images/source/architecture-overview.excalidraw new file mode 100644 index 0000000..3a43dd2 --- /dev/null +++ b/docs/images/source/architecture-overview.excalidraw @@ -0,0 +1,4918 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "rectangle", + "version": 725, + "versionNonce": 1177834708, + "index": "a0", + "isDeleted": false, + "id": "0kT__OqPgnIQGYnuSJQEy", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0.0007550076501088299, + "x": 631.3128420932196, + "y": -455.9073302807943, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 953.452931543973, + "height": 901.1269208777178, + "seed": 1899513620, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 269, + "versionNonce": 629609068, + "index": "a2", + "isDeleted": false, + "id": "wz9oH0zDCSZ2T-tPqOmMT", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 679.0404671395848, + "y": -494.36784794178936, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 121.13990294933319, + "height": 25, + "seed": 2025487532, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "K8s Cluster", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "K8s Cluster", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 171, + "versionNonce": 670734932, + "index": "a3", + "isDeleted": false, + "id": "AirwiUeLUlZoJ35J1Y5qd", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 797.79877719342, + "y": -250.028818213237, + "strokeColor": "#000000", + "backgroundColor": "#a5d8ff", + "width": 229.49400491361894, + "height": 134.93853507796484, + "seed": 641048596, + "groupIds": [ + "07rmWrX_nO8N_9KrvJv0r" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "UThCCvcOgZhsv9pzBe9-t", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 293, + "versionNonce": 1582425324, + "index": "a4", + "isDeleted": false, + "id": "NhpiSS3jifrJTYNquP31D", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 801.2461120311783, + "y": -246.58148337547885, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 222.5993352381026, + "height": 44.32287648546298, + "seed": 1150238612, + "groupIds": [ + "07rmWrX_nO8N_9KrvJv0r" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 226, + "versionNonce": 1406939092, + "index": "a5", + "isDeleted": false, + "id": "kfaht_ng2eFExlxy6tbMJ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 819.4677390307576, + "y": -235.25452605141612, + "strokeColor": "#1e1e1e", + "backgroundColor": "#1971c2", + "width": 188, + "height": 21.507525856732585, + "seed": 1157390740, + "groupIds": [ + "07rmWrX_nO8N_9KrvJv0r" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 17.20602068538607, + "fontFamily": 1, + "text": "Namespace A (tenant)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Namespace A (tenant)", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 262, + "versionNonce": 1266839829, + "index": "a6", + "isDeleted": false, + "id": "VO3y3f4Bro6k2TzJIFIqq", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 797.3063007880264, + "y": 90.27237791403866, + "strokeColor": "#000000", + "backgroundColor": "#a5d8ff", + "width": 229.49400491361894, + "height": 134.93853507796484, + "seed": 725271084, + "groupIds": [ + "Odncg0IquQdLRWrfSxFj7" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "tK2oSaYZ27UjkamNMTtm8", + "type": "arrow" + } + ], + "updated": 1726574306491, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 389, + "versionNonce": 1797882293, + "index": "a7", + "isDeleted": false, + "id": "Vv7mYZNi5TQLs8EdticoP", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 800.7536356257847, + "y": 93.7197127517968, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 222.5993352381026, + "height": 44.32287648546298, + "seed": 427400364, + "groupIds": [ + "Odncg0IquQdLRWrfSxFj7" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "X1SVfZoGpSutYjwq_QAoS", + "type": "arrow" + } + ], + "updated": 1726573032724, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 323, + "versionNonce": 1466982427, + "index": "a8", + "isDeleted": false, + "id": "8u4PP56wDGANbELO37YyY", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 818.9752626253639, + "y": 105.04667007585954, + "strokeColor": "#1e1e1e", + "backgroundColor": "#1971c2", + "width": 189.21665954589844, + "height": 21.507525856732585, + "seed": 1073293100, + "groupIds": [ + "Odncg0IquQdLRWrfSxFj7" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 17.20602068538607, + "fontFamily": 1, + "text": "Namespace B (tenant)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Namespace B (tenant)", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "line", + "version": 1576, + "versionNonce": 186541932, + "index": "a9", + "isDeleted": false, + "id": "1i3xhYIYgSO6UegTTQIi-", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 851.7524051217313, + "y": -194.68947743096925, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 1278169811, + "groupIds": [ + "VCG3yShAN59gr1dOiHwdt", + "E0bmNeXWMY2H08bS91xoD" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 915, + "versionNonce": 166663508, + "index": "aA", + "isDeleted": false, + "id": "Xpr_1b82KfSWLmHzqzUYR", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 851.0583649722344, + "y": -195.41649280535884, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 1728282739, + "groupIds": [ + "VCG3yShAN59gr1dOiHwdt", + "E0bmNeXWMY2H08bS91xoD" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 612, + "versionNonce": 540449260, + "index": "aB", + "isDeleted": false, + "id": "6eArdZVcGx04BWnOLUSHI", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 839.0617074890806, + "y": -172.69610675372917, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 11.606398134643495, + "height": 20.43580530455634, + "seed": 507574803, + "groupIds": [ + "KLieHSjMGmJCKq0xyoW1Q", + "E0bmNeXWMY2H08bS91xoD" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.10680734479733643, + 14.205376858045264 + ], + [ + 11.321578548517282, + 20.43580530455634 + ], + [ + 11.606398134643495, + 3.8806668609697628 + ], + [ + 1.3528930340995515, + 1.032470999707551 + ] + ] + }, + { + "type": "line", + "version": 830, + "versionNonce": 394873556, + "index": "aC", + "isDeleted": false, + "id": "vSV9mXbloQ31AORpuiQJW", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 864.4818555508459, + "y": -172.7673116502607, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 12.667312471753975, + "height": 20.509145060610187, + "seed": 786317235, + "groupIds": [ + "KLieHSjMGmJCKq0xyoW1Q", + "E0bmNeXWMY2H08bS91xoD" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -12.667312471753975, + 3.6435687166298947 + ], + [ + -11.887638165338293, + 20.509145060610187 + ], + [ + -0.5696391722524426, + 14.347786651108377 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 545, + "versionNonce": 2024045676, + "index": "aD", + "isDeleted": false, + "id": "9NRb7N6MiKbxRG-U-Q0dp", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 839.9873711439909, + "y": -174.33381937395492, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 24.458881958589213, + "height": 7.5121165840790765, + "seed": 1929498963, + "groupIds": [ + "KLieHSjMGmJCKq0xyoW1Q", + "E0bmNeXWMY2H08bS91xoD" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.962422617301264, + 3.4890399300462054 + ], + [ + 24.458881958589213, + 0.28481958612622416 + ], + [ + 11.855615272503934, + -4.023076654032871 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 1674, + "versionNonce": 814215675, + "index": "aE", + "isDeleted": false, + "id": "CL_SfWXM4dxNGFmWO_Pyf", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 853.5120092134973, + "y": 149.79386171720938, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 905855357, + "groupIds": [ + "HpUkH1owie5Y9mShtJW0t", + "dduBkEwooUDC_DmIiL8JG" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 1013, + "versionNonce": 1906605877, + "index": "aF", + "isDeleted": false, + "id": "RKIpd0ovpmG4zzbFbPoKu", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 852.8179690640004, + "y": 149.06684634281982, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 252905949, + "groupIds": [ + "HpUkH1owie5Y9mShtJW0t", + "dduBkEwooUDC_DmIiL8JG" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 710, + "versionNonce": 1213069979, + "index": "aG", + "isDeleted": false, + "id": "yV_KforUE0R8isLXQjRqC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 840.8213115808467, + "y": 171.78723239444952, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 11.606398134643495, + "height": 20.43580530455634, + "seed": 254789181, + "groupIds": [ + "GmrD5Cd3RUsmwl3lkjcYm", + "dduBkEwooUDC_DmIiL8JG" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.10680734479733643, + 14.205376858045264 + ], + [ + 11.321578548517282, + 20.43580530455634 + ], + [ + 11.606398134643495, + 3.8806668609697628 + ], + [ + 1.3528930340995515, + 1.032470999707551 + ] + ] + }, + { + "type": "line", + "version": 928, + "versionNonce": 664326293, + "index": "aH", + "isDeleted": false, + "id": "PBzr46-UYQYvYNLAfYqNM", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 866.241459642612, + "y": 171.716027497918, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 12.667312471753975, + "height": 20.509145060610187, + "seed": 2033579677, + "groupIds": [ + "GmrD5Cd3RUsmwl3lkjcYm", + "dduBkEwooUDC_DmIiL8JG" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -12.667312471753975, + 3.6435687166298947 + ], + [ + -11.887638165338293, + 20.509145060610187 + ], + [ + -0.5696391722524426, + 14.347786651108377 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 643, + "versionNonce": 1468657467, + "index": "aI", + "isDeleted": false, + "id": "Bx8JYJkHqtd2MVW88cF5d", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 841.746975235757, + "y": 170.14951977422373, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 24.458881958589213, + "height": 7.5121165840790765, + "seed": 936844029, + "groupIds": [ + "GmrD5Cd3RUsmwl3lkjcYm", + "dduBkEwooUDC_DmIiL8JG" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.962422617301264, + 3.4890399300462054 + ], + [ + 24.458881958589213, + 0.28481958612622416 + ], + [ + 11.855615272503934, + -4.023076654032871 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "rectangle", + "version": 520, + "versionNonce": 792776788, + "index": "aJ", + "isDeleted": false, + "id": "IXM-ycKKLynDI9_se0tbS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 828.8506738722192, + "y": -57.93817526959725, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 1689191539, + "groupIds": [ + "F0HHYTpkvzerjK3snodKc", + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "X1SVfZoGpSutYjwq_QAoS", + "type": "arrow" + }, + { + "id": "BcQt_9XVaxScWMFDPUWj6", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 659, + "versionNonce": 2117209836, + "index": "aK", + "isDeleted": false, + "id": "K0PQqlJLSiQr-m_nNRMp3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 831.1889391720464, + "y": -55.59990996977018, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 1675361811, + "groupIds": [ + "F0HHYTpkvzerjK3snodKc", + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 669, + "versionNonce": 1089185236, + "index": "aL", + "isDeleted": false, + "id": "x8EXGmX1rbEbx59OuJDqp", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 855.6491056347087, + "y": -47.91703827033823, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 309904307, + "groupIds": [ + "F0HHYTpkvzerjK3snodKc", + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "H0xq397VZEaqEZd8VIuVa", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 280, + "versionNonce": 1365856620, + "index": "aM", + "isDeleted": false, + "id": "FUCJ84ZqclMliSpVlh6PD", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 867.7854951881153, + "y": -4.5318171283138895, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 81.4398865495705, + "height": 23.617567099375446, + "seed": 282102589, + "groupIds": [ + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0d73e5101f3f34bbd383fd4890b4a5b193c4cd7d", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "arrow", + "version": 297, + "versionNonce": 838122677, + "index": "aN", + "isDeleted": false, + "id": "X1SVfZoGpSutYjwq_QAoS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 903.9714725003059, + "y": 35.13240249435863, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 4.128856931323526, + "height": 51.68264046758965, + "seed": 1180273580, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": { + "elementId": "IXM-ycKKLynDI9_se0tbS", + "focus": 0.07963690087316273, + "gap": 1.5441931707229486, + "fixedPoint": null + }, + "endBinding": { + "elementId": "Vv7mYZNi5TQLs8EdticoP", + "focus": -0.01442398254926684, + "gap": 6.904669789848526, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": "triangle", + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 4.128856931323526, + 51.68264046758965 + ] + ] + }, + { + "type": "arrow", + "version": 290, + "versionNonce": 1369426772, + "index": "aO", + "isDeleted": false, + "id": "H0xq397VZEaqEZd8VIuVa", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 907.77738763787, + "y": -57.57410773164113, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 0.03332855851738259, + "height": 57.72688970644844, + "seed": 1338406572, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": { + "elementId": "x8EXGmX1rbEbx59OuJDqp", + "focus": 0.009046945851759977, + "gap": 9.657069461302887, + "fixedPoint": null + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": "triangle", + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -0.03332855851738259, + -57.72688970644844 + ] + ] + }, + { + "type": "text", + "version": 209, + "versionNonce": 1905542124, + "index": "aP", + "isDeleted": false, + "id": "OP5vo21WkKJxLDQ7WsaIP", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 928.2226562248311, + "y": -101.45743302712805, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 43.95000076293945, + "height": 25, + "seed": 909529492, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "H0xq397VZEaqEZd8VIuVa", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 263, + "versionNonce": 370604475, + "index": "aQ", + "isDeleted": false, + "id": "oHXSqIs_4r9-kqPP9K6LL", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 929.1455605188958, + "y": 52.16533990440996, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 43.95000076293945, + "height": 25, + "seed": 1245528364, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 682, + "versionNonce": 1404256468, + "index": "aR", + "isDeleted": false, + "id": "YUIfnB0Ep0feA1SxOCbrb", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1115.0906552509139, + "y": -69.81385212236, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 1659332989, + "groupIds": [ + "e2hvUYTvDhDfyLe7jQUrP", + "qwJPGORBFXuwBY4vMNji2", + "dXx6-WSoCt6ziqSeiYSVU" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "BcQt_9XVaxScWMFDPUWj6", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 823, + "versionNonce": 1690101356, + "index": "aS", + "isDeleted": false, + "id": "vh0kMt4uRu2qNT5p6K9Rm", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1117.428920550741, + "y": -67.47558682253293, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 345453021, + "groupIds": [ + "e2hvUYTvDhDfyLe7jQUrP", + "qwJPGORBFXuwBY4vMNji2", + "dXx6-WSoCt6ziqSeiYSVU" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "HRC4nvQnImi7JrVTnPffO", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 829, + "versionNonce": 2126251604, + "index": "aT", + "isDeleted": false, + "id": "KVoFpRmBld0LRg-d7VUw5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1141.8890870134032, + "y": -59.792715123100976, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 569306685, + "groupIds": [ + "e2hvUYTvDhDfyLe7jQUrP", + "qwJPGORBFXuwBY4vMNji2", + "dXx6-WSoCt6ziqSeiYSVU" + ], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "HRC4nvQnImi7JrVTnPffO", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 216, + "versionNonce": 1605187820, + "index": "aU", + "isDeleted": false, + "id": "K-OW_9PihXZAinCZtxtl7", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1149.530068566248, + "y": -15.93013107137233, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 74.78507893498988, + "height": 24.36673143007783, + "seed": 196836500, + "groupIds": [ + "dXx6-WSoCt6ziqSeiYSVU" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "status": "saved", + "fileId": "89e65ecfb11f47f117431eaf415737057f89540a", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "arrow", + "version": 342, + "versionNonce": 663755732, + "index": "aV", + "isDeleted": false, + "id": "BcQt_9XVaxScWMFDPUWj6", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 987.113828368865, + "y": -2.0540240239824503, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 117.20884534614072, + "height": 2.4235620298425045, + "seed": 1283133100, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": { + "elementId": "IXM-ycKKLynDI9_se0tbS", + "focus": 0.24811077165866896, + "gap": 2.6014931081547275, + "fixedPoint": null + }, + "endBinding": { + "elementId": "YUIfnB0Ep0feA1SxOCbrb", + "focus": -0.37450168545181567, + "gap": 10.767981535908234, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 117.20884534614072, + -2.4235620298425045 + ] + ] + }, + { + "type": "text", + "version": 247, + "versionNonce": 1844394860, + "index": "aW", + "isDeleted": false, + "id": "WdDTr4tLaV40gzp27fXwN", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1021.4359899253054, + "y": -40.12508950199981, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 43.95000076293945, + "height": 25, + "seed": 1769858092, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 609, + "versionNonce": 407208276, + "index": "aX", + "isDeleted": false, + "id": "nQSiJMErGetnI9kXYB_hn", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2064.699917067238, + "y": -389.2986609539827, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 109.62509643488907, + "height": 109.62509643488907, + "seed": 1699992492, + "groupIds": [ + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "rectangle", + "version": 634, + "versionNonce": 432860652, + "index": "aY", + "isDeleted": false, + "id": "yP5auBfAXJK7hUfWl-Vjd", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1685.6006357892197, + "y": -287.20342502436733, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 543.8990028571536, + "height": 520.0550840960977, + "seed": 821638700, + "groupIds": [ + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 383, + "versionNonce": 1094745717, + "index": "aZ", + "isDeleted": false, + "id": "AvqIyGOl5Gf2B9rlRr89y", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1768.9140845306397, + "y": 11.98675453387142, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 404.2320808000759, + "height": 122.7462711105253, + "seed": 736282131, + "groupIds": [ + "JAoyv5ZJ797_hQWk9epJo", + "gu5aCopAaFQ6Nr6B-dVBB", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "rCjEpau87ivsN4t1tqcRY", + "type": "arrow" + }, + { + "id": "sXRPVPb6f6tvxn9cVVxfk", + "type": "arrow" + }, + { + "id": "JfsG1-wNhMg3_3Vme50iY", + "type": "arrow" + } + ], + "updated": 1726574215100, + "link": null, + "locked": false + }, + { + "type": "image", + "version": 571, + "versionNonce": 1157045237, + "index": "aa", + "isDeleted": false, + "id": "3zI2MqgaDE2KUupshfpDs", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1833.2654960805557, + "y": 42.76461766526114, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 1046907827, + "groupIds": [ + "JAoyv5ZJ797_hQWk9epJo", + "gu5aCopAaFQ6Nr6B-dVBB", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "image", + "version": 480, + "versionNonce": 1475990869, + "index": "ab", + "isDeleted": false, + "id": "xOLHBYIJGs560qJ3luN6h", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1944.590699332588, + "y": 44.764617665261085, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 154.68292682926847, + "height": 61.00000000000006, + "seed": 60857683, + "groupIds": [ + "JAoyv5ZJ797_hQWk9epJo", + "gu5aCopAaFQ6Nr6B-dVBB", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "status": "saved", + "fileId": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "text", + "version": 192, + "versionNonce": 1629381301, + "index": "ac", + "isDeleted": false, + "id": "fw-V9HrBursFwxZmXLfcp", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1783.9967921059706, + "y": -22.64010991086593, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 90.06666564941406, + "height": 25, + "seed": 427310131, + "groupIds": [ + "gu5aCopAaFQ6Nr6B-dVBB", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "tenant B", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "tenant B", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 370, + "versionNonce": 1748121300, + "index": "ad", + "isDeleted": false, + "id": "ZeSBFPHSIRJrNIVONV8pd", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1757.012046714095, + "y": -212.41797304225565, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 404.2320808000759, + "height": 122.7462711105253, + "seed": 2134047788, + "groupIds": [ + "FvHRSAHHvVveAz-zUiixI", + "dECj0ihkJyQZq3o1rW4Q2", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "XgVSqQJ4AuvikC6lnCvPL", + "type": "arrow" + }, + { + "id": "QBgO3jI5Xi_tY6nUzWyog", + "type": "arrow" + }, + { + "id": "-QST69y_eZ_BHi_2u1R2e", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "image", + "version": 538, + "versionNonce": 56693141, + "index": "ae", + "isDeleted": false, + "id": "23mLq92mHzbQylngpAduQ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1821.3634582640111, + "y": -181.64010991086593, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 516997491, + "groupIds": [ + "FvHRSAHHvVveAz-zUiixI", + "dECj0ihkJyQZq3o1rW4Q2", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "image", + "version": 485, + "versionNonce": 340807788, + "index": "af", + "isDeleted": false, + "id": "n7NA_AGQqIM_tucZFPfjC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1932.6886615160433, + "y": -179.640109910866, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 154.68292682926847, + "height": 61.00000000000006, + "seed": 933320563, + "groupIds": [ + "FvHRSAHHvVveAz-zUiixI", + "dECj0ihkJyQZq3o1rW4Q2", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "status": "saved", + "fileId": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "text", + "version": 301, + "versionNonce": 1407990868, + "index": "ag", + "isDeleted": false, + "id": "rPho_sRNRiDOZWwS0UmrX", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1765.7051241677382, + "y": -247.14010991086593, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 88.6500015258789, + "height": 25, + "seed": 939876979, + "groupIds": [ + "dECj0ihkJyQZq3o1rW4Q2", + "j13myXqeT-ig5dpdEKlbf" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "tenant A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 1295, + "versionNonce": 1564697324, + "index": "ah", + "isDeleted": false, + "id": "6jeTkacRGdJjX2shicWIe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1373.2179451973516, + "y": -214.83677442505763, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 960297747, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "pPdRo0KtbJuRokRMIcoiB", + "type": "arrow" + }, + { + "id": "HRC4nvQnImi7JrVTnPffO", + "type": "arrow" + } + ], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1438, + "versionNonce": 357142996, + "index": "ai", + "isDeleted": false, + "id": "A-6ZZNjW4FYYmnJP700J3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1375.5562104971787, + "y": -212.49850912523056, + "strokeColor": "#a5d8ff", + "backgroundColor": "#a5d8ff", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 1823613107, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1478, + "versionNonce": 1699582956, + "index": "aj", + "isDeleted": false, + "id": "gWYN6qI5kHy9Ul-OBNTGi", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1386.5375332708759, + "y": -204.81563742579857, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 130.30709838867188, + "height": 29.176336945926444, + "seed": 1840353875, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044142402, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "observability-operator \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "observability-operator \nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 708, + "versionNonce": 2120105452, + "index": "ak", + "isDeleted": false, + "id": "B0INGCHiq2u1yQA-uaNuV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1400.5528118413038, + "y": -168.57358212844116, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 106.99192810058594, + "height": 40, + "seed": 596782003, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044146606, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Observability \nOperator*", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Observability \nOperator*", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 1028, + "versionNonce": 62035637, + "index": "al", + "isDeleted": false, + "id": "2BtNaoyJrv53rOHCPQikx", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1103.1992942364325, + "y": -257.20846469933326, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 141.52638459323293, + "seed": 536455389, + "groupIds": [ + "MAmCff9sLMHrWd-tTf5r-", + "8juRk7vTiG3pwOsPozvnb" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "8tFLVP9bF6wpqhD4Nc3Eb", + "type": "arrow" + }, + { + "id": "pPdRo0KtbJuRokRMIcoiB", + "type": "arrow" + } + ], + "updated": 1726574056530, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1125, + "versionNonce": 1035481269, + "index": "am", + "isDeleted": false, + "id": "dAkcRVhV8V6-w0XtBXnIj", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1105.5375595362596, + "y": -254.8701993995062, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 669680957, + "groupIds": [ + "MAmCff9sLMHrWd-tTf5r-", + "8juRk7vTiG3pwOsPozvnb" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "QBgO3jI5Xi_tY6nUzWyog", + "type": "arrow" + } + ], + "updated": 1726574090036, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1030, + "versionNonce": 1625284716, + "index": "an", + "isDeleted": false, + "id": "gy1FJQLe96vpeaKtD_cMP", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1132.9977259989216, + "y": -249.1873277000742, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 1245762973, + "groupIds": [ + "8juRk7vTiG3pwOsPozvnb" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737043867787, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 578, + "versionNonce": 262576443, + "index": "ao", + "isDeleted": false, + "id": "MD1jQdbQflmtABlNXVPYh", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1142.816509907203, + "y": -205.1899970993038, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 2082660403, + "groupIds": [ + "j9NjwVQAMhnXlqYZ5BkUu", + "ot5JCU1Kt38iE-6fzoSj2", + "_LlYFed0-lJxd6QYQselq" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "rectangle", + "version": 1134, + "versionNonce": 378989557, + "index": "ap", + "isDeleted": false, + "id": "aNsoOPZhkhKKo9MvXHF8b", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1108.199294236432, + "y": 77.54681060407972, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 141.52638459323293, + "seed": 124540637, + "groupIds": [ + "8BJAXrFEEycKK1ualfLV0", + "ZrwRRipDen2r0FYfR5xqV" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "yOrB4EPZ6qPM_oO8OTRig", + "type": "arrow" + }, + { + "id": "_32Ato2xOBIvtu1trLMfe", + "type": "arrow" + }, + { + "id": "sXRPVPb6f6tvxn9cVVxfk", + "type": "arrow" + } + ], + "updated": 1726573032724, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1230, + "versionNonce": 1939192283, + "index": "aq", + "isDeleted": false, + "id": "sR0NhZ_cU6uaURVFiSSkM", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1110.537559536259, + "y": 79.8850759039068, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 685208381, + "groupIds": [ + "8BJAXrFEEycKK1ualfLV0", + "ZrwRRipDen2r0FYfR5xqV" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1136, + "versionNonce": 1231445972, + "index": "ar", + "isDeleted": false, + "id": "hQJqvw-o5D4GkiPugpbUr", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1137.9977259989216, + "y": 85.56794760333878, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 996148125, + "groupIds": [ + "ZrwRRipDen2r0FYfR5xqV" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737043938701, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 646, + "versionNonce": 437765755, + "index": "as", + "isDeleted": false, + "id": "ydmVrZ_g9oxk5y4FnFlFO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1148.8165099072035, + "y": 135.8100029006962, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 1515204285, + "groupIds": [ + "ac9ZstwZhdzYRIXgicbHc", + "FhokNDeTVk5F-pk3c6Knw", + "_1PGniXJZbRHmUY00TAMY" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "arrow", + "version": 128, + "versionNonce": 251131925, + "index": "at", + "isDeleted": false, + "id": "8tFLVP9bF6wpqhD4Nc3Eb", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1102.0301249306776, + "y": -173.79391291605424, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 220, + "height": 17.10391581675043, + "seed": 1478065235, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574056532, + "link": null, + "locked": false, + "startBinding": { + "elementId": "2BtNaoyJrv53rOHCPQikx", + "focus": -0.08474347907800667, + "gap": 1.1691693057549628, + "fixedPoint": null + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -220, + 17.10391581675043 + ] + ] + }, + { + "type": "text", + "version": 102, + "versionNonce": 166674203, + "index": "au", + "isDeleted": false, + "id": "To-7wajg2FH7V9_7ILTLY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 895.7301256936171, + "y": -193.6899970993038, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 122.5999984741211, + "height": 20, + "seed": 856105907, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "scrapes metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "scrapes metrics", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 110, + "versionNonce": 1056707605, + "index": "av", + "isDeleted": false, + "id": "yOrB4EPZ6qPM_oO8OTRig", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1107.0301249306776, + "y": 173.3100029006962, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 222, + "height": 7, + "seed": 540911155, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "startBinding": { + "elementId": "aNsoOPZhkhKKo9MvXHF8b", + "focus": -0.30742739218377424, + "gap": 1.169169305754508, + "fixedPoint": null + }, + "endBinding": { + "elementId": "wigFb16qo2dTFho4ODNO2", + "focus": 1.4725755946790995, + "gap": 15.700000762939453, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -222, + 7 + ] + ] + }, + { + "type": "text", + "version": 182, + "versionNonce": 1301193659, + "index": "aw", + "isDeleted": false, + "id": "wigFb16qo2dTFho4ODNO2", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 900.7301256936171, + "y": 150.3100029006962, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 122.5999984741211, + "height": 20, + "seed": 211653021, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "yOrB4EPZ6qPM_oO8OTRig", + "type": "arrow" + } + ], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "scrapes metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "scrapes metrics", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 1352, + "versionNonce": 542913781, + "index": "ax", + "isDeleted": false, + "id": "kkYs_p-VIQQ6bYRq6zBN_", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1376.199294236432, + "y": 99.57660478113758, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 694234835, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "_32Ato2xOBIvtu1trLMfe", + "type": "arrow" + }, + { + "id": "bSS3DPyvWAxZsfir-Pty0", + "type": "arrow" + } + ], + "updated": 1726574366310, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1490, + "versionNonce": 667080085, + "index": "ay", + "isDeleted": false, + "id": "QlgbWdG6JVD8X1Ia9A_vi", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1377.537559536259, + "y": 101.91487008096465, + "strokeColor": "#a5d8ff", + "backgroundColor": "#a5d8ff", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 2040126579, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1726573844655, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1506, + "versionNonce": 1554854124, + "index": "az", + "isDeleted": false, + "id": "2-cSdTgw9XYr1ZHVv7UlA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1387.7944564745362, + "y": 109.46366798363647, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 130.30709838867188, + "height": 29.176336945926444, + "seed": 1168889363, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044153236, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "observability-operator \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "observability-operator \nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 749, + "versionNonce": 2002498772, + "index": "b00", + "isDeleted": false, + "id": "1FTFKr-5knTNjgdx-qZ3g", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1402.4091608803844, + "y": 145.83979707775404, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 106.99192810058594, + "height": 40, + "seed": 1552784307, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044157104, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Observability \nOperator*", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Observability \nOperator*", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 87, + "versionNonce": 1398839276, + "index": "b01", + "isDeleted": false, + "id": "HxepbWNN1JLlmX5zuhogx", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1268.0301249306776, + "y": -7.68999709930381, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 490, + "height": 115, + "seed": 517444627, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 490, + -115 + ] + ] + }, + { + "type": "arrow", + "version": 71, + "versionNonce": 2045962197, + "index": "b02", + "isDeleted": false, + "id": "rCjEpau87ivsN4t1tqcRY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1267.0301249306776, + "y": -1.6899970993038096, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 498, + "height": 68.03787153950307, + "seed": 301460147, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "AvqIyGOl5Gf2B9rlRr89y", + "focus": -0.23747558753985967, + "gap": 3.883959599962054, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 498, + 68.03787153950307 + ] + ] + }, + { + "type": "text", + "version": 84, + "versionNonce": 770367893, + "index": "b03", + "isDeleted": false, + "id": "cIgJaHwyaqg98JVTqSQPo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1137.5051234047987, + "y": -142.6899970993038, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 103.05000305175781, + "height": 20, + "seed": 1656536765, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "prometheus-A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "prometheus-A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 130, + "versionNonce": 601714235, + "index": "b04", + "isDeleted": false, + "id": "u_OGpLO6vZ2uEZNoCgint", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1138.9384577553847, + "y": 196.3100029006962, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 104.18333435058594, + "height": 20, + "seed": 1579375325, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "prometheus-B", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "prometheus-B", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 354, + "versionNonce": 42347732, + "index": "b05", + "isDeleted": false, + "id": "HzdNdxiQ_PKDvVR_VeFEC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.0702469576001885, + "x": 1379.0301249306776, + "y": -92.18999709930381, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 182, + "height": 25, + "seed": 410503709, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs for tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs for tenant A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 506, + "versionNonce": 1680469723, + "index": "b06", + "isDeleted": false, + "id": "tElSewQ-shroM6SKQKfkY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0.10487693873023396, + "x": 1371.3217890542128, + "y": 41.81000290069619, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 183.4166717529297, + "height": 25, + "seed": 865053661, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs for tenant B", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs for tenant B", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 72, + "versionNonce": 929602235, + "index": "b07", + "isDeleted": false, + "id": "_32Ato2xOBIvtu1trLMfe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1370.0301249306776, + "y": 158.3391953720155, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 102, + "height": 4.040662408103628, + "seed": 1564650931, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726573831484, + "link": null, + "locked": false, + "startBinding": { + "elementId": "kkYs_p-VIQQ6bYRq6zBN_", + "focus": -0.3342518506124808, + "gap": 6.169169305754508, + "fixedPoint": null + }, + "endBinding": { + "elementId": "aNsoOPZhkhKKo9MvXHF8b", + "focus": 0.03727338113728061, + "gap": 4.169169305754508, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -102, + -4.040662408103628 + ] + ] + }, + { + "type": "arrow", + "version": 327, + "versionNonce": 1100358741, + "index": "b08", + "isDeleted": false, + "id": "pPdRo0KtbJuRokRMIcoiB", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1349.3854832597285, + "y": -167.1715305494588, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 86.35535832905089, + "height": 3.7033035001409758, + "seed": 61142003, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574057490, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "2BtNaoyJrv53rOHCPQikx", + "focus": 0.1626688454202745, + "gap": 4.169169305754053, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -86.35535832905089, + -3.7033035001409758 + ] + ] + }, + { + "type": "text", + "version": 57, + "versionNonce": 1330933173, + "index": "b09", + "isDeleted": false, + "id": "0WE83i5fnLWD90YnSofrZ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1295.7801249306776, + "y": 124.31000290069619, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 66.5, + "height": 20, + "seed": 562734547, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Manages", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Manages", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 119, + "versionNonce": 1905735067, + "index": "b0A", + "isDeleted": false, + "id": "SpmxfZvplgGhas98SIMvy", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1289.0248496272648, + "y": -200.21527891009015, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 66.5, + "height": 20, + "seed": 178914237, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574057485, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Manages", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Manages", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 215, + "versionNonce": 1910422124, + "index": "b0B", + "isDeleted": false, + "id": "HRC4nvQnImi7JrVTnPffO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1370.0487758915965, + "y": -142.49168376565578, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 100.63472455201872, + "height": 78.88609762330569, + "seed": 692113619, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": { + "elementId": "6jeTkacRGdJjX2shicWIe", + "focus": 0.34486732617819776, + "gap": 3.169169305754963, + "fixedPoint": null + }, + "endBinding": { + "elementId": "vh0kMt4uRu2qNT5p6K9Rm", + "focus": 0.567366277678479, + "gap": 1, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -100.63472455201872, + 78.88609762330569 + ] + ] + }, + { + "type": "text", + "version": 282, + "versionNonce": 743535188, + "index": "b0C", + "isDeleted": false, + "id": "7b7XRWD5ZPdReGFKcRSeB", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.67060997096873, + "x": 1268.0976310822061, + "y": -127.4874748300212, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 79.33333587646484, + "height": 20, + "seed": 2100696883, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Configures", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Configures", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 343, + "versionNonce": 882537269, + "index": "b0D", + "isDeleted": false, + "id": "3rAQUgrWF6XPeSswdMjgI", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 659.7121975416102, + "y": 473.40245388444964, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 479.5, + "height": 40, + "seed": 1525909085, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574400551, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "* Just one instance. \nDuplicated to avoid arrows crossing each other in the image", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "* Just one instance. \nDuplicated to avoid arrows crossing each other in the image", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 285, + "versionNonce": 928064821, + "index": "b0E", + "isDeleted": false, + "id": "sXRPVPb6f6tvxn9cVVxfk", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1270.714594361458, + "y": 205.60129841265268, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 637.2856452737662, + "height": 233.95835576991044, + "seed": 1334603251, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "startBinding": { + "elementId": "aNsoOPZhkhKKo9MvXHF8b", + "focus": 0.13468597409221764, + "gap": 6.85363873653489, + "fixedPoint": null + }, + "endBinding": { + "elementId": "AvqIyGOl5Gf2B9rlRr89y", + "focus": -0.08067957051648472, + "gap": 4.947751037584965, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 334.830943381892, + 168.03783403923944 + ], + [ + 637.2856452737662, + -65.920521730671 + ] + ] + }, + { + "type": "text", + "version": 458, + "versionNonce": 1454108117, + "index": "b0F", + "isDeleted": false, + "id": "d7NHTFFLmM5EEZztBdd-w", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.256875589926665, + "x": 1480.1799456497167, + "y": 292.67927413960956, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 210.89999389648438, + "height": 25, + "seed": 1724657437, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Metrics for tenant B", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Metrics for tenant B", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 145, + "versionNonce": 308150997, + "index": "b0G", + "isDeleted": false, + "id": "QBgO3jI5Xi_tY6nUzWyog", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1264.490970878523, + "y": -236.27596887571895, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 670.9312615627969, + "height": 158.12992365498167, + "seed": 531849203, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574215101, + "link": null, + "locked": false, + "startBinding": { + "elementId": "dAkcRVhV8V6-w0XtBXnIj", + "focus": 0.7008811292253831, + "gap": 7.968280553426553, + "fixedPoint": null + }, + "endBinding": { + "elementId": "ZeSBFPHSIRJrNIVONV8pd", + "focus": 0.3701488995551903, + "gap": 2.647788803047206, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 293.75502839452247, + -136.91971662456558 + ], + [ + 670.9312615627969, + 21.210207030416086 + ] + ] + }, + { + "type": "text", + "version": 543, + "versionNonce": 1660513077, + "index": "b0H", + "isDeleted": false, + "id": "lxbDChugwhOjw0A3LrO4t", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.256875589926665, + "x": 1426.1203872469607, + "y": -339.64087172656696, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 209.48333740234375, + "height": 25, + "seed": 949017085, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726573032724, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Metrics for tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Metrics for tenant A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 265, + "versionNonce": 292850924, + "index": "b0k", + "isDeleted": false, + "id": "Pyr0oxIz6PMyDk-Vach3U", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2426.9687904908637, + "y": -135.49873371016247, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 74.75, + "height": 20, + "seed": 1993372947, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Tenant A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 330, + "versionNonce": 2044972244, + "index": "b0l", + "isDeleted": false, + "id": "rTYYnP2RiJ7e-PVQWRYnP", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2428.833017473173, + "y": 176.41779400834088, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 75.88333129882812, + "height": 20, + "seed": 2049810099, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737043835690, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Tenant B", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Tenant B", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 76, + "versionNonce": 1866707028, + "index": "b0m", + "isDeleted": false, + "id": "JfsG1-wNhMg3_3Vme50iY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2419.724895020224, + "y": 121.74543179429452, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 234.2804770599596, + "height": 17.78379293981351, + "seed": 1382034707, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737043833132, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "AvqIyGOl5Gf2B9rlRr89y", + "focus": 0.18674168779063785, + "gap": 12.298252629548642, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -234.2804770599596, + -17.78379293981351 + ] + ] + }, + { + "type": "arrow", + "version": 124, + "versionNonce": 1469581268, + "index": "b0n", + "isDeleted": false, + "id": "XgVSqQJ4AuvikC6lnCvPL", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2408.6190155786326, + "y": -159.07905824581974, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 243.3339707464138, + "height": 6.276987426922403, + "seed": 1380568243, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "ZeSBFPHSIRJrNIVONV8pd", + "focus": 0.05347516896028915, + "gap": 4.040917318047832, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -243.3339707464138, + 6.276987426922403 + ] + ] + }, + { + "type": "text", + "version": 388, + "versionNonce": 330043244, + "index": "b0o", + "isDeleted": false, + "id": "51hvrX_heXRjBZbZUo3uV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1882.3132909945914, + "y": 250.79506751598984, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 354.7833251953125, + "height": 40, + "seed": 630992829, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "** It is also valid for other remote writes, \nsuch as Victoria Metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "** It is also valid for other remote writes, \nsuch as Victoria Metrics", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 154, + "versionNonce": 1483730260, + "index": "b0p", + "isDeleted": false, + "id": "dpv0hndxVEnOPNFQZHS7i", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2160.29397998097, + "y": -380.39576295886883, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 41.21485204073218, + "height": 49.95739641300868, + "seed": 1245208947, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737044133704, + "link": null, + "locked": false, + "fontSize": 39.96591713040696, + "fontFamily": 1, + "text": "**", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "**", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "wxEWxzR2Ymt0Q4qKrNZct", + "type": "image", + "x": 1156.2090224043482, + "y": -372.65799823588077, + "width": 57.4803918930104, + "height": 58.16708666615247, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#a5d8ff", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0q", + "roundness": null, + "seed": 669006971, + "version": 580, + "versionNonce": 1878588731, + "isDeleted": false, + "boundElements": [], + "updated": 1726574033787, + "link": null, + "locked": false, + "status": "saved", + "fileId": "2a37a629f89b41313a2ed687420bfef7b4c588d3", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "rectangle", + "version": 1094, + "versionNonce": 1017666389, + "index": "b0r", + "isDeleted": false, + "id": "8B8n349m8iKlCc_zqjvjR", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1104.9370553820845, + "y": -427.45318347602125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 141.52638459323293, + "seed": 292243413, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "LCtg3Sz9HMeQ3hHChQI_p", + "type": "arrow" + }, + { + "id": "UThCCvcOgZhsv9pzBe9-t", + "type": "arrow" + }, + { + "id": "-QST69y_eZ_BHi_2u1R2e", + "type": "arrow" + } + ], + "updated": 1726574208726, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1190, + "versionNonce": 1469288731, + "index": "b0s", + "isDeleted": false, + "id": "OubIh89F3TEWjggxNgjvt", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1107.2753206819116, + "y": -425.1149181761942, + "strokeColor": "#f08c00", + "backgroundColor": "#f08c00", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 12420405, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1726574024574, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1105, + "versionNonce": 457083244, + "index": "b0t", + "isDeleted": false, + "id": "ASpag-A6TXm-9tOhR8hH0", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1151.3175533326446, + "y": -419.9746540297586, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 70.83684864640236, + "height": 29.176336945926444, + "seed": 966858389, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737043872678, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "observability\nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "observability\nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 166, + "versionNonce": 1324115707, + "index": "b0u", + "isDeleted": false, + "id": "kWgJLG8k7owwIoBtZ5Dbl", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1150.2318881323447, + "y": -311.6518385093829, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 69.16796374320984, + "height": 20, + "seed": 1639517877, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574044236, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "otelcol-A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "otelcol-A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 442, + "versionNonce": 1411204213, + "index": "b0v", + "isDeleted": false, + "id": "LCtg3Sz9HMeQ3hHChQI_p", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1382.7684226868787, + "y": -238.17083295218583, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 90.35242132562053, + "height": 57.69570371440187, + "seed": 1616750709, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574070802, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "8B8n349m8iKlCc_zqjvjR", + "focus": -0.07632017991924242, + "gap": 31.81728459068279, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -90.35242132562053, + -57.69570371440187 + ] + ] + }, + { + "type": "text", + "version": 303, + "versionNonce": 1562209819, + "index": "b0w", + "isDeleted": false, + "id": "WMpOjt6-fXBh1hin9NUn1", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0.48775556639861506, + "x": 1339.707791955376, + "y": -270.54840414672213, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 66.5, + "height": 20, + "seed": 1339740629, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574076390, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Manages", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Manages", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 325, + "versionNonce": 1707999893, + "index": "b0x", + "isDeleted": false, + "id": "UThCCvcOgZhsv9pzBe9-t", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 982.3382089768503, + "y": -266.9416613426618, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 106.0252482386494, + "height": 55.99328704185206, + "seed": 1712021, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574116941, + "link": null, + "locked": false, + "startBinding": { + "elementId": "AirwiUeLUlZoJ35J1Y5qd", + "focus": -0.3710810362047797, + "gap": 16.9128431294248, + "fixedPoint": null + }, + "endBinding": { + "elementId": "8B8n349m8iKlCc_zqjvjR", + "focus": 0.14393156089495177, + "gap": 16.573598166584702, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 106.0252482386494, + -55.99328704185206 + ] + ] + }, + { + "type": "text", + "version": 262, + "versionNonce": 602977237, + "index": "b0y", + "isDeleted": false, + "id": "SkrWiS1MgHKphhNeh3wey", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.934937477788751, + "x": 949.8767211493689, + "y": -318.3338374209683, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 112.55732727050781, + "height": 22.1534636342351, + "seed": 1596203349, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574243599, + "link": null, + "locked": false, + "fontSize": 17.722770907388078, + "fontFamily": 1, + "text": "push traces ", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "push traces ", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "vuUyMa33uefvaejc0lJP-", + "type": "image", + "x": 2088.8539442199976, + "y": -174.29427200749888, + "width": 58.38744075619255, + "height": 47.59845713820045, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#f08c00", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0z", + "roundness": null, + "seed": 109913883, + "version": 224, + "versionNonce": 1308415963, + "isDeleted": false, + "boundElements": [], + "updated": 1726574178257, + "link": null, + "locked": false, + "status": "saved", + "fileId": "2c7e0abafe59feadb391c72f782764d488654533", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "image", + "version": 271, + "versionNonce": 444977141, + "index": "b11", + "isDeleted": false, + "id": "lOVJF4ZgQ75oB142hz5Dl", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2095.931301068155, + "y": 50.090100023851505, + "strokeColor": "transparent", + "backgroundColor": "#f08c00", + "width": 58.38744075619255, + "height": 47.59845713820045, + "seed": 799464731, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574182408, + "link": null, + "locked": false, + "status": "saved", + "fileId": "2c7e0abafe59feadb391c72f782764d488654533", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "arrow", + "version": 677, + "versionNonce": 475177973, + "index": "b12", + "isDeleted": false, + "id": "-QST69y_eZ_BHi_2u1R2e", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1276.8934243786464, + "y": -339.4367100056738, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 647.6028318397091, + "height": 169.2117427701564, + "seed": 1965389845, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574230017, + "link": null, + "locked": false, + "startBinding": { + "elementId": "8B8n349m8iKlCc_zqjvjR", + "focus": 0.3948117331577366, + "gap": 16.294707608071008, + "fixedPoint": null + }, + "endBinding": { + "elementId": "ZeSBFPHSIRJrNIVONV8pd", + "focus": 0.2950562617187041, + "gap": 30.515203482079954, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 431.4033513406091, + -72.70820928881824 + ], + [ + 647.6028318397091, + 96.50353348133817 + ] + ] + }, + { + "type": "text", + "version": 426, + "versionNonce": 734744475, + "index": "b14", + "isDeleted": false, + "id": "V0Yy2XZSBdg_q9k4Cuf9V", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.134894885016612, + "x": 1387.7543715213162, + "y": -415.13992658467316, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 184.90804278850555, + "height": 22.1534636342351, + "seed": 78826517, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574344143, + "link": null, + "locked": false, + "fontSize": 17.722770907388078, + "fontFamily": 1, + "text": "Traces for tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Traces for tenant A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 654, + "versionNonce": 945698043, + "index": "b15", + "isDeleted": false, + "id": "DhLaeEZ1NG2XniCA6AGCs", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1163.1190155784884, + "y": 316.72358335508346, + "strokeColor": "transparent", + "backgroundColor": "#a5d8ff", + "width": 57.4803918930104, + "height": 58.16708666615247, + "seed": 2091425429, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574274885, + "link": null, + "locked": false, + "status": "saved", + "fileId": "2a37a629f89b41313a2ed687420bfef7b4c588d3", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "rectangle", + "version": 1170, + "versionNonce": 1438560891, + "index": "b16", + "isDeleted": false, + "id": "iHGsRu2inQGPh7Fr88TmJ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1111.8470485562248, + "y": 261.9283981149429, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 141.52638459323293, + "seed": 799146997, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "tK2oSaYZ27UjkamNMTtm8", + "type": "arrow" + }, + { + "id": "QjPnq2qfIhukLXWNzHbOd", + "type": "arrow" + } + ], + "updated": 1726574318514, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1264, + "versionNonce": 1525379643, + "index": "b17", + "isDeleted": false, + "id": "xFYWk39kJY64tl_SdHc3C", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1114.1853138560518, + "y": 264.26666341477, + "strokeColor": "#f08c00", + "backgroundColor": "#f08c00", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 2097659221, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1726574274885, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1175, + "versionNonce": 5249748, + "index": "b18", + "isDeleted": false, + "id": "Nw5HEwPh2tO1Gf15I0HBT", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1158.2275465067846, + "y": 269.4069275612056, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 70.83684864640236, + "height": 29.176336945926444, + "seed": 2083474101, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1737043900765, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "observability\nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "observability\nNamespace", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 240, + "versionNonce": 985292667, + "index": "b19", + "isDeleted": false, + "id": "re0khFdokM17-kfFL2Ip5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1157.1418813064852, + "y": 377.7297430815814, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 69.16796374320984, + "height": 20, + "seed": 1035873301, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574274885, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "otelcol-A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "otelcol-A", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 612, + "versionNonce": 432370293, + "index": "b1A", + "isDeleted": false, + "id": "tK2oSaYZ27UjkamNMTtm8", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 935.2196091969929, + "y": 246.66903342421944, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 146.06803149145367, + "height": 84.78004419843654, + "seed": 1414179285, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574308702, + "link": null, + "locked": false, + "startBinding": { + "elementId": "VO3y3f4Bro6k2TzJIFIqq", + "focus": 0.5629988218107009, + "gap": 21.458120432215935, + "fixedPoint": null + }, + "endBinding": { + "elementId": "iHGsRu2inQGPh7Fr88TmJ", + "focus": -0.5319146936079019, + "gap": 30.559407867778077, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 146.06803149145367, + 84.78004419843654 + ] + ] + }, + { + "type": "text", + "version": 390, + "versionNonce": 2079254011, + "index": "b1B", + "isDeleted": false, + "id": "cvP5z2eg9KP9anRyFOAiZ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0.46705575887496487, + "x": 923.7564704361553, + "y": 292.10789010383616, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 112.55732727050781, + "height": 22.1534636342351, + "seed": 895526709, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574304028, + "link": null, + "locked": false, + "fontSize": 17.722770907388078, + "fontFamily": 1, + "text": "push traces ", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "push traces ", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 917, + "versionNonce": 12783899, + "index": "b1C", + "isDeleted": false, + "id": "QjPnq2qfIhukLXWNzHbOd", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1279.716639550575, + "y": 352.055099972456, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 627.5107666963534, + "height": 236.00525984792318, + "seed": 1231163381, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574324014, + "link": null, + "locked": false, + "startBinding": { + "elementId": "iHGsRu2inQGPh7Fr88TmJ", + "focus": 0.09591160097922753, + "gap": 12.207929605859476, + "fixedPoint": null + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 408.1226634029324, + 52.641666192759544 + ], + [ + 627.5107666963534, + -183.36359365516364 + ] + ] + }, + { + "type": "text", + "version": 580, + "versionNonce": 1207973147, + "index": "b1D", + "isDeleted": false, + "id": "PzvbAyZzIeuQ7u7YAhLbB", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0.1543651580020935, + "x": 1363.9606885480339, + "y": 362.023481229124, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 186.04212176799774, + "height": 22.1534636342351, + "seed": 1693098197, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574341286, + "link": null, + "locked": false, + "fontSize": 17.722770907388078, + "fontFamily": 1, + "text": "Traces for tenant B", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Traces for tenant B", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 275, + "versionNonce": 1072047387, + "index": "b1E", + "isDeleted": false, + "id": "bSS3DPyvWAxZsfir-Pty0", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.486748351761064, + "x": 1439.3534404910463, + "y": 271.3710790483273, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 156.27779921120532, + "height": 19.7445912631525, + "seed": 1442130459, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1726574368437, + "link": null, + "locked": false, + "startBinding": { + "elementId": "kkYs_p-VIQQ6bYRq6zBN_", + "focus": -0.4009675554707781, + "gap": 21.989429512027442, + "fixedPoint": null + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -156.27779921120532, + -19.7445912631525 + ] + ] + }, + { + "type": "text", + "version": 286, + "versionNonce": 1173902293, + "index": "b1F", + "isDeleted": false, + "id": "2EzpLTNcKdlydu2iGWyq0", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.631544261315902, + "x": 1342.7659656373032, + "y": 216.77594889474597, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 66.5, + "height": 20, + "seed": 396308155, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1726574370826, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Manages", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Manages", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "line", + "version": 1320, + "versionNonce": 653988436, + "isDeleted": false, + "id": "qz78NNGbbatH5EyIhfqxn", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2443.859132515879, + "y": 155.41954132251658, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 49.26942071813747, + "height": 43.87300421060919, + "seed": 2084870740, + "groupIds": [ + "oIUxAbjN4b8R7gBX1-gkE", + "se9LqlHQqOY-3KL26PYnK" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1737043835690, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.518175120431392, + -29.072472669680792 + ], + [ + 23.649321944705985, + -43.87300421060919 + ], + [ + 41.780468768980576, + -32.244015142736835 + ], + [ + 49.26942071813747, + -3.0188078795298896 + ], + [ + 0, + 0 + ] + ], + "index": "b1H", + "frameId": null, + "roundness": { + "type": 2 + } + }, + { + "type": "ellipse", + "version": 1057, + "versionNonce": 202221524, + "isDeleted": false, + "id": "Cb6txN4x2t6xnjnjwCChD", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2456.3130033521948, + "y": 88.72849353666015, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 25.225943407686408, + "height": 22.072700481725683, + "seed": 249966548, + "groupIds": [ + "oIUxAbjN4b8R7gBX1-gkE", + "se9LqlHQqOY-3KL26PYnK" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1737043835690, + "link": null, + "locked": false, + "index": "b1I", + "frameId": null, + "roundness": null + }, + { + "type": "line", + "version": 1365, + "versionNonce": 1830606060, + "isDeleted": false, + "id": "s91guUwEnZLAlUdQ-IiIh", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2437.734132515879, + "y": -152.01439510673822, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 49.26942071813747, + "height": 43.87300421060919, + "seed": 1723330028, + "groupIds": [ + "s9JlFvkq2kmEgg7azmxt0", + "VWOyfcy-Ol3Xhdq_YCVnp" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1737043843656, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.518175120431392, + -29.072472669680792 + ], + [ + 23.649321944705985, + -43.87300421060919 + ], + [ + 41.780468768980576, + -32.244015142736835 + ], + [ + 49.26942071813747, + -3.0188078795298896 + ], + [ + 0, + 0 + ] + ], + "index": "b1J", + "frameId": null, + "roundness": { + "type": 2 + } + }, + { + "type": "ellipse", + "version": 1102, + "versionNonce": 1661687660, + "isDeleted": false, + "id": "rur3g1dlg-wyAEnoqly_I", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2450.1880033521943, + "y": -218.70544289259465, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 25.225943407686408, + "height": 22.072700481725683, + "seed": 198922348, + "groupIds": [ + "s9JlFvkq2kmEgg7azmxt0", + "VWOyfcy-Ol3Xhdq_YCVnp" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1737043843656, + "link": null, + "locked": false, + "index": "b1K", + "frameId": null, + "roundness": null + } + ], + "appState": { + "gridSize": 20, + "gridStep": 5, + "gridModeEnabled": false, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "0d73e5101f3f34bbd383fd4890b4a5b193c4cd7d": { + "mimeType": "image/png", + "id": "0d73e5101f3f34bbd383fd4890b4a5b193c4cd7d", + "dataURL": "", + "created": 1726574587460, + "lastRetrieved": 1737043646801 + }, + "89e65ecfb11f47f117431eaf415737057f89540a": { + "mimeType": "image/png", + "id": "89e65ecfb11f47f117431eaf415737057f89540a", + "dataURL": "", + "created": 1726574587462, + "lastRetrieved": 1737043646801 + }, + "0644a8702c31871ef24e831da4f3ed873aa2d825": { + "mimeType": "image/png", + "id": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "dataURL": "", + "created": 1726574587466, + "lastRetrieved": 1737043646801 + }, + "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78": { + "mimeType": "image/png", + "id": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "dataURL": "", + "created": 1726574587467, + "lastRetrieved": 1737043646801 + }, + "1493a0629beacbb7e4599ef52f315c4c7b9f2307": { + "mimeType": "image/png", + "id": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "dataURL": "", + "created": 1726574587474, + "lastRetrieved": 1737043646801 + }, + "2a37a629f89b41313a2ed687420bfef7b4c588d3": { + "mimeType": "image/png", + "id": "2a37a629f89b41313a2ed687420bfef7b4c588d3", + "dataURL": "", + "created": 1726574587497, + "lastRetrieved": 1737043646801 + }, + "2c7e0abafe59feadb391c72f782764d488654533": { + "mimeType": "image/png", + "id": "2c7e0abafe59feadb391c72f782764d488654533", + "dataURL": "", + "created": 1726574587516, + "lastRetrieved": 1737043646801 + } + } +} \ No newline at end of file diff --git a/docs/images/source/logs-overview.excalidraw b/docs/images/source/logs-overview.excalidraw new file mode 100644 index 0000000..4d8ec25 --- /dev/null +++ b/docs/images/source/logs-overview.excalidraw @@ -0,0 +1,1790 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "rectangle", + "version": 727, + "versionNonce": 965446636, + "isDeleted": false, + "id": "0kT__OqPgnIQGYnuSJQEy", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 633.1110041900688, + "y": -283.4038015734899, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 953.4529315439731, + "height": 336.9239992715183, + "seed": 1899513620, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "a0", + "frameId": null + }, + { + "type": "text", + "version": 317, + "versionNonce": 827205204, + "isDeleted": false, + "id": "wz9oH0zDCSZ2T-tPqOmMT", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 685.6316148877967, + "y": -323.3776961841621, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 121.13990294933319, + "height": 25, + "seed": 2025487532, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044233639, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "K8s Cluster", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "K8s Cluster", + "lineHeight": 1.25, + "baseline": 18, + "index": "a2", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 173, + "versionNonce": 634797652, + "isDeleted": false, + "id": "AirwiUeLUlZoJ35J1Y5qd", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 797.79877719342, + "y": -250.028818213237, + "strokeColor": "#000000", + "backgroundColor": "#a5d8ff", + "width": 229.49400491361894, + "height": 134.93853507796484, + "seed": 641048596, + "groupIds": [ + "07rmWrX_nO8N_9KrvJv0r" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "a3", + "frameId": null + }, + { + "type": "rectangle", + "version": 295, + "versionNonce": 2020770028, + "isDeleted": false, + "id": "NhpiSS3jifrJTYNquP31D", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 801.2461120311783, + "y": -246.58148337547885, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 222.5993352381026, + "height": 44.32287648546298, + "seed": 1150238612, + "groupIds": [ + "07rmWrX_nO8N_9KrvJv0r" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "a4", + "frameId": null + }, + { + "type": "text", + "version": 228, + "versionNonce": 1587762132, + "isDeleted": false, + "id": "kfaht_ng2eFExlxy6tbMJ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 819.4677390307576, + "y": -235.25452605141612, + "strokeColor": "#1e1e1e", + "backgroundColor": "#1971c2", + "width": 188, + "height": 21.507525856732585, + "seed": 1157390740, + "groupIds": [ + "07rmWrX_nO8N_9KrvJv0r" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 17.20602068538607, + "fontFamily": 1, + "text": "Namespace A (tenant)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Namespace A (tenant)", + "lineHeight": 1.25, + "baseline": 15, + "index": "a5", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1578, + "versionNonce": 1321960300, + "isDeleted": false, + "id": "1i3xhYIYgSO6UegTTQIi-", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 851.7524051217313, + "y": -194.68947743096925, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 1278169811, + "groupIds": [ + "VCG3yShAN59gr1dOiHwdt", + "E0bmNeXWMY2H08bS91xoD" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ], + "index": "a6", + "frameId": null + }, + { + "type": "line", + "version": 917, + "versionNonce": 2016524628, + "isDeleted": false, + "id": "Xpr_1b82KfSWLmHzqzUYR", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 851.0583649722344, + "y": -195.41649280535884, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 1728282739, + "groupIds": [ + "VCG3yShAN59gr1dOiHwdt", + "E0bmNeXWMY2H08bS91xoD" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ], + "index": "a7", + "frameId": null + }, + { + "type": "line", + "version": 614, + "versionNonce": 669005292, + "isDeleted": false, + "id": "6eArdZVcGx04BWnOLUSHI", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 839.0617074890806, + "y": -172.69610675372917, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 11.606398134643495, + "height": 20.43580530455634, + "seed": 507574803, + "groupIds": [ + "KLieHSjMGmJCKq0xyoW1Q", + "E0bmNeXWMY2H08bS91xoD" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.10680734479733643, + 14.205376858045264 + ], + [ + 11.321578548517282, + 20.43580530455634 + ], + [ + 11.606398134643495, + 3.8806668609697628 + ], + [ + 1.3528930340995515, + 1.032470999707551 + ] + ], + "index": "a8", + "frameId": null + }, + { + "type": "line", + "version": 832, + "versionNonce": 1842835156, + "isDeleted": false, + "id": "vSV9mXbloQ31AORpuiQJW", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 864.4818555508459, + "y": -172.7673116502607, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 12.667312471753975, + "height": 20.509145060610187, + "seed": 786317235, + "groupIds": [ + "KLieHSjMGmJCKq0xyoW1Q", + "E0bmNeXWMY2H08bS91xoD" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -12.667312471753975, + 3.6435687166298947 + ], + [ + -11.887638165338293, + 20.509145060610187 + ], + [ + -0.5696391722524426, + 14.347786651108377 + ], + [ + 0, + 0 + ] + ], + "index": "a9", + "frameId": null + }, + { + "type": "line", + "version": 547, + "versionNonce": 280664172, + "isDeleted": false, + "id": "9NRb7N6MiKbxRG-U-Q0dp", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 839.9873711439909, + "y": -174.33381937395492, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 24.458881958589213, + "height": 7.5121165840790765, + "seed": 1929498963, + "groupIds": [ + "KLieHSjMGmJCKq0xyoW1Q", + "E0bmNeXWMY2H08bS91xoD" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.962422617301264, + 3.4890399300462054 + ], + [ + 24.458881958589213, + 0.28481958612622416 + ], + [ + 11.855615272503934, + -4.023076654032871 + ], + [ + 0, + 0 + ] + ], + "index": "aA", + "frameId": null + }, + { + "type": "rectangle", + "version": 523, + "versionNonce": 175427052, + "isDeleted": false, + "id": "IXM-ycKKLynDI9_se0tbS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 828.8506738722192, + "y": -58.06317526959725, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 1689191539, + "groupIds": [ + "F0HHYTpkvzerjK3snodKc", + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "BcQt_9XVaxScWMFDPUWj6", + "type": "arrow" + } + ], + "updated": 1737044236251, + "link": null, + "locked": false, + "index": "aB", + "frameId": null + }, + { + "type": "rectangle", + "version": 662, + "versionNonce": 1836270316, + "isDeleted": false, + "id": "K0PQqlJLSiQr-m_nNRMp3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 831.1889391720464, + "y": -55.72490996977018, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 1675361811, + "groupIds": [ + "F0HHYTpkvzerjK3snodKc", + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044236252, + "link": null, + "locked": false, + "index": "aC", + "frameId": null + }, + { + "type": "text", + "version": 677, + "versionNonce": 341301356, + "isDeleted": false, + "id": "x8EXGmX1rbEbx59OuJDqp", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 855.6491056347087, + "y": -48.04203827033823, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 309904307, + "groupIds": [ + "F0HHYTpkvzerjK3snodKc", + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "roundness": null, + "boundElements": [ + { + "id": "H0xq397VZEaqEZd8VIuVa", + "type": "arrow" + } + ], + "updated": 1737044241013, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aD", + "frameId": null, + "autoResize": true + }, + { + "type": "image", + "version": 283, + "versionNonce": 496671340, + "isDeleted": false, + "id": "FUCJ84ZqclMliSpVlh6PD", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 867.7854951881153, + "y": -4.6568171283138895, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 81.4398865495705, + "height": 23.617567099375446, + "seed": 282102589, + "groupIds": [ + "2qAhoff8_WZ8Q1ZJ8Sjuz" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044236252, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0d73e5101f3f34bbd383fd4890b4a5b193c4cd7d", + "scale": [ + 1, + 1 + ], + "index": "aE", + "frameId": null, + "crop": null + }, + { + "type": "arrow", + "version": 297, + "versionNonce": 52030188, + "isDeleted": false, + "id": "H0xq397VZEaqEZd8VIuVa", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 907.7773663416891, + "y": -57.69910773164112, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 0.03330726233650694, + "height": 57.601889706448446, + "seed": 1338406572, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044241014, + "link": null, + "locked": false, + "startBinding": { + "elementId": "x8EXGmX1rbEbx59OuJDqp", + "focus": 0.009046945851759977, + "gap": 9.657069461302887 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": "triangle", + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -0.03330726233650694, + -57.601889706448446 + ] + ], + "index": "aF", + "frameId": null + }, + { + "type": "text", + "version": 211, + "versionNonce": 965730284, + "isDeleted": false, + "id": "OP5vo21WkKJxLDQ7WsaIP", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 928.2226562248311, + "y": -101.45743302712805, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 43.95000076293945, + "height": 25, + "seed": 909529492, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "H0xq397VZEaqEZd8VIuVa", + "type": "arrow" + } + ], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs", + "lineHeight": 1.25, + "baseline": 18, + "index": "aG", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 685, + "versionNonce": 1781164628, + "isDeleted": false, + "id": "YUIfnB0Ep0feA1SxOCbrb", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1114.9656552509139, + "y": -69.82166462236, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 1659332989, + "groupIds": [ + "e2hvUYTvDhDfyLe7jQUrP", + "qwJPGORBFXuwBY4vMNji2", + "dXx6-WSoCt6ziqSeiYSVU" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "BcQt_9XVaxScWMFDPUWj6", + "type": "arrow" + } + ], + "updated": 1737044243749, + "link": null, + "locked": false, + "index": "aH", + "frameId": null + }, + { + "type": "rectangle", + "version": 826, + "versionNonce": 1047192916, + "isDeleted": false, + "id": "vh0kMt4uRu2qNT5p6K9Rm", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1117.303920550741, + "y": -67.48339932253293, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 345453021, + "groupIds": [ + "e2hvUYTvDhDfyLe7jQUrP", + "qwJPGORBFXuwBY4vMNji2", + "dXx6-WSoCt6ziqSeiYSVU" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "HRC4nvQnImi7JrVTnPffO", + "type": "arrow" + } + ], + "updated": 1737044243749, + "link": null, + "locked": false, + "index": "aI", + "frameId": null + }, + { + "type": "text", + "version": 837, + "versionNonce": 920915028, + "isDeleted": false, + "id": "KVoFpRmBld0LRg-d7VUw5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1141.7640870134032, + "y": -59.800527623100976, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 569306685, + "groupIds": [ + "e2hvUYTvDhDfyLe7jQUrP", + "qwJPGORBFXuwBY4vMNji2", + "dXx6-WSoCt6ziqSeiYSVU" + ], + "roundness": null, + "boundElements": [ + { + "id": "HRC4nvQnImi7JrVTnPffO", + "type": "arrow" + } + ], + "updated": 1737044247433, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aJ", + "frameId": null, + "autoResize": true + }, + { + "type": "image", + "version": 218, + "versionNonce": 443667692, + "isDeleted": false, + "id": "K-OW_9PihXZAinCZtxtl7", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1149.530068566248, + "y": -15.81294357137233, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 74.78507893498988, + "height": 24.36673143007783, + "seed": 196836500, + "groupIds": [ + "dXx6-WSoCt6ziqSeiYSVU" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "status": "saved", + "fileId": "89e65ecfb11f47f117431eaf415737057f89540a", + "scale": [ + 1, + 1 + ], + "index": "aK", + "frameId": null, + "crop": null + }, + { + "type": "arrow", + "version": 346, + "versionNonce": 1603403732, + "isDeleted": false, + "id": "BcQt_9XVaxScWMFDPUWj6", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 987.1138283688649, + "y": -2.136156917290612, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 117.08384534614083, + "height": 2.385210729328569, + "seed": 1283133100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044243749, + "link": null, + "locked": false, + "startBinding": { + "elementId": "IXM-ycKKLynDI9_se0tbS", + "focus": 0.24811077165866896, + "gap": 2.6014931081547275 + }, + "endBinding": { + "elementId": "YUIfnB0Ep0feA1SxOCbrb", + "focus": -0.37450168545181567, + "gap": 10.767981535908234 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 117.08384534614083, + -2.385210729328569 + ] + ], + "index": "aL", + "frameId": null + }, + { + "type": "text", + "version": 249, + "versionNonce": 499144556, + "isDeleted": false, + "id": "WdDTr4tLaV40gzp27fXwN", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1021.4359899253054, + "y": -40.12508950199981, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 43.95000076293945, + "height": 25, + "seed": 1769858092, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs", + "lineHeight": 1.25, + "baseline": 18, + "index": "aM", + "frameId": null, + "autoResize": true + }, + { + "type": "image", + "version": 611, + "versionNonce": 1587815764, + "isDeleted": false, + "id": "nQSiJMErGetnI9kXYB_hn", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1687.74816396492, + "y": -391.1652732546999, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 109.62509643488907, + "height": 109.62509643488907, + "seed": 1699992492, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "scale": [ + 1, + 1 + ], + "index": "aN", + "frameId": null, + "crop": null + }, + { + "type": "rectangle", + "version": 636, + "versionNonce": 1763346924, + "isDeleted": false, + "id": "yP5auBfAXJK7hUfWl-Vjd", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1685.6006357892197, + "y": -287.25331221280516, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 543.8990028571536, + "height": 341.73162633557274, + "seed": 821638700, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "aO", + "frameId": null + }, + { + "type": "rectangle", + "version": 372, + "versionNonce": 857496276, + "isDeleted": false, + "id": "ZeSBFPHSIRJrNIVONV8pd", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1756.012046714095, + "y": -211.46786023069347, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 404.2320808000759, + "height": 122.7462711105253, + "seed": 2134047788, + "groupIds": [ + "dECj0ihkJyQZq3o1rW4Q2" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "XgVSqQJ4AuvikC6lnCvPL", + "type": "arrow" + } + ], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "aP", + "frameId": null + }, + { + "type": "image", + "version": 487, + "versionNonce": 1602907244, + "isDeleted": false, + "id": "n7NA_AGQqIM_tucZFPfjC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1887.6886615160433, + "y": -178.6899970993038, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 154.68292682926847, + "height": 61.00000000000006, + "seed": 933320563, + "groupIds": [ + "dECj0ihkJyQZq3o1rW4Q2" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "status": "saved", + "fileId": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "scale": [ + 1, + 1 + ], + "index": "aQ", + "frameId": null, + "crop": null + }, + { + "type": "text", + "version": 303, + "versionNonce": 99737684, + "isDeleted": false, + "id": "rPho_sRNRiDOZWwS0UmrX", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1764.7051241677382, + "y": -246.18999709930375, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 88.6500015258789, + "height": 25, + "seed": 939876979, + "groupIds": [ + "dECj0ihkJyQZq3o1rW4Q2" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "tenant A", + "lineHeight": 1.25, + "baseline": 18, + "index": "aR", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 1297, + "versionNonce": 1926889196, + "isDeleted": false, + "id": "6jeTkacRGdJjX2shicWIe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1371.1992942364325, + "y": -214.80342016076804, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 960297747, + "groupIds": [ + "ulaWA2luSAPfvq9xo4IP4" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "HRC4nvQnImi7JrVTnPffO", + "type": "arrow" + } + ], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "aS", + "frameId": null + }, + { + "type": "rectangle", + "version": 1440, + "versionNonce": 1056617940, + "isDeleted": false, + "id": "A-6ZZNjW4FYYmnJP700J3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1373.5375595362596, + "y": -212.46515486094097, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 1823613107, + "groupIds": [ + "ulaWA2luSAPfvq9xo4IP4" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "index": "aT", + "frameId": null + }, + { + "type": "text", + "version": 1485, + "versionNonce": 1469282388, + "isDeleted": false, + "id": "gWYN6qI5kHy9Ul-OBNTGi", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1397.9977259989216, + "y": -204.78228316150899, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 1840353875, + "groupIds": [ + "ulaWA2luSAPfvq9xo4IP4" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044254383, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aU", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 728, + "versionNonce": 702003436, + "isDeleted": false, + "id": "B0INGCHiq2u1yQA-uaNuV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1402.2701609414198, + "y": -168.54022786415157, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 99.51992797851562, + "height": 40, + "seed": 596782003, + "groupIds": [ + "ulaWA2luSAPfvq9xo4IP4" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044262872, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Observability\n Operator*", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Observability\n Operator*", + "lineHeight": 1.25, + "baseline": 34, + "index": "aV", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 89, + "versionNonce": 1895838700, + "isDeleted": false, + "id": "HxepbWNN1JLlmX5zuhogx", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1268.0301249306776, + "y": -7.68999709930381, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 490, + "height": 115, + "seed": 517444627, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 490, + -115 + ] + ], + "index": "aW", + "frameId": null + }, + { + "type": "text", + "version": 356, + "versionNonce": 1161711828, + "isDeleted": false, + "id": "HzdNdxiQ_PKDvVR_VeFEC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.0702469576001885, + "x": 1379.0301249306776, + "y": -92.18999709930381, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 182, + "height": 25, + "seed": 410503709, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Logs for tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Logs for tenant A", + "lineHeight": 1.25, + "baseline": 18, + "index": "aX", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 218, + "versionNonce": 1812667092, + "isDeleted": false, + "id": "HRC4nvQnImi7JrVTnPffO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1368.0301249306776, + "y": -141.91039062932924, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 98.74107359109985, + "height": 78.06937893522588, + "seed": 692113619, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044243749, + "link": null, + "locked": false, + "startBinding": { + "elementId": "6jeTkacRGdJjX2shicWIe", + "focus": 0.34486732617819776, + "gap": 3.169169305754963 + }, + "endBinding": { + "elementId": "vh0kMt4uRu2qNT5p6K9Rm", + "focus": 0.567366277678479, + "gap": 1 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -98.74107359109985, + 78.06937893522588 + ] + ], + "index": "aY", + "frameId": null + }, + { + "type": "text", + "version": 284, + "versionNonce": 744822356, + "isDeleted": false, + "id": "7b7XRWD5ZPdReGFKcRSeB", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.67060997096873, + "x": 1268.0976310822061, + "y": -127.4874748300212, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 79.33333587646484, + "height": 20, + "seed": 2100696883, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Configures", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Configures", + "lineHeight": 1.25, + "baseline": 14, + "index": "aZ", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 314, + "versionNonce": 1296849492, + "isDeleted": false, + "id": "Pyr0oxIz6PMyDk-Vach3U", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2554.9062904908637, + "y": -145.03779621016247, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 74.75, + "height": 20, + "seed": 1993372947, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044281974, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Tenant A", + "lineHeight": 1.25, + "baseline": 14, + "index": "ao", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 127, + "versionNonce": 105131348, + "isDeleted": false, + "id": "XgVSqQJ4AuvikC6lnCvPL", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 2533.4588593286326, + "y": -178.30050786098255, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 369.1738144964138, + "height": 26.167944015647493, + "seed": 1380568243, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044275617, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "ZeSBFPHSIRJrNIVONV8pd", + "focus": 0.16611802797687547, + "gap": 4.040917318047832 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -369.1738144964138, + 26.167944015647493 + ] + ], + "index": "ap", + "frameId": null + }, + { + "type": "text", + "version": 390, + "versionNonce": 397704428, + "isDeleted": false, + "id": "51hvrX_heXRjBZbZUo3uV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1879.4965947933174, + "y": 68.48974802990381, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 354.7833251953125, + "height": 40, + "seed": 630992829, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "** It is also valid for other remote writes, \nsuch as Victoria Metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "** It is also valid for other remote writes, \nsuch as Victoria Metrics", + "lineHeight": 1.25, + "baseline": 34, + "index": "aq", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 156, + "versionNonce": 1234114516, + "isDeleted": false, + "id": "dpv0hndxVEnOPNFQZHS7i", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1774.5620817135855, + "y": -385.19324758031155, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 41.21485204073218, + "height": 49.95739641300868, + "seed": 1245208947, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044225668, + "link": null, + "locked": false, + "fontSize": 39.96591713040696, + "fontFamily": 1, + "text": "**", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "**", + "lineHeight": 1.25, + "baseline": 35, + "index": "ar", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1315, + "versionNonce": 1783237204, + "isDeleted": false, + "id": "1T2jdVrKHKLd-4yI9G-Bh", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2563.578973112788, + "y": -162.9853807829487, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 49.26942071813747, + "height": 43.87300421060919, + "seed": 689900780, + "groupIds": [ + "ETF4UYmPONVNlNrjT2bRG", + "5XJfdVuTmCLwAcaMEmANS" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1737044283191, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.518175120431392, + -29.072472669680792 + ], + [ + 23.649321944705985, + -43.87300421060919 + ], + [ + 41.780468768980576, + -32.244015142736835 + ], + [ + 49.26942071813747, + -3.0188078795298896 + ], + [ + 0, + 0 + ] + ], + "index": "at", + "frameId": null, + "roundness": { + "type": 2 + } + }, + { + "type": "ellipse", + "version": 1052, + "versionNonce": 2059247572, + "isDeleted": false, + "id": "IOzMxXHcBpgrJR0gJVK8v", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2576.0328439491036, + "y": -229.67642856880514, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 25.225943407686408, + "height": 22.072700481725683, + "seed": 1376161644, + "groupIds": [ + "ETF4UYmPONVNlNrjT2bRG", + "5XJfdVuTmCLwAcaMEmANS" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1737044283191, + "link": null, + "locked": false, + "index": "au", + "frameId": null, + "roundness": null + } + ], + "appState": { + "gridSize": 20, + "gridStep": 5, + "gridModeEnabled": false, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "0d73e5101f3f34bbd383fd4890b4a5b193c4cd7d": { + "mimeType": "image/png", + "id": "0d73e5101f3f34bbd383fd4890b4a5b193c4cd7d", + "dataURL": "", + "created": 1726574587460, + "lastRetrieved": 1737043646801 + }, + "89e65ecfb11f47f117431eaf415737057f89540a": { + "mimeType": "image/png", + "id": "89e65ecfb11f47f117431eaf415737057f89540a", + "dataURL": "", + "created": 1726574587462, + "lastRetrieved": 1737043646801 + }, + "0644a8702c31871ef24e831da4f3ed873aa2d825": { + "mimeType": "image/png", + "id": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "dataURL": "", + "created": 1726574587466, + "lastRetrieved": 1737043646801 + }, + "1493a0629beacbb7e4599ef52f315c4c7b9f2307": { + "mimeType": "image/png", + "id": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "dataURL": "", + "created": 1726574587474, + "lastRetrieved": 1737043646801 + } + } +} \ No newline at end of file diff --git a/docs/images/source/send-data-different-backend.excalidraw b/docs/images/source/send-data-different-backend.excalidraw new file mode 100644 index 0000000..79fb0d8 --- /dev/null +++ b/docs/images/source/send-data-different-backend.excalidraw @@ -0,0 +1,6225 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "image", + "version": 1115, + "versionNonce": 744478956, + "isDeleted": false, + "id": "51P_ipCxt3SGjchRBgDYJ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1501.9398304379815, + "y": -17.24562932449834, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 47.29465917562221, + "height": 47.29465917562221, + "seed": 1668416691, + "groupIds": [ + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "scale": [ + 1, + 1 + ], + "index": "Za", + "frameId": null, + "crop": null + }, + { + "type": "rectangle", + "version": 1177, + "versionNonce": 1263393644, + "isDeleted": false, + "id": "qCLTpBrh4o5Fh5Rs7SoIk", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1501.0133399130668, + "y": 27.58425855309963, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 234.64990045749352, + "height": 115.17659860618852, + "seed": 1255470675, + "groupIds": [ + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "index": "Zb", + "frameId": null + }, + { + "type": "rectangle", + "version": 894, + "versionNonce": 307727852, + "isDeleted": false, + "id": "I7fko55wggHsMjE_Zjhfo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1531.8217811439326, + "y": 59.848334734684386, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 174.3945420440032, + "height": 52.955420301032916, + "seed": 1479062515, + "groupIds": [ + "yKiV6gPTanKGtvKyuBZkk", + "hKl7RjmIcNyVk0DtQRD7k", + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "e-8ZsFtNd0YWIHs4OXK9I", + "type": "arrow" + } + ], + "updated": 1737044445859, + "link": null, + "locked": false, + "index": "Zc", + "frameId": null + }, + { + "type": "image", + "version": 1082, + "versionNonce": 1806041836, + "isDeleted": false, + "id": "2Fc9c2IqFS8qhO9nP84CT", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1559.584384559576, + "y": 73.12657661807998, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 33.83521872075687, + "height": 28.042418628487233, + "seed": 729005459, + "groupIds": [ + "yKiV6gPTanKGtvKyuBZkk", + "hKl7RjmIcNyVk0DtQRD7k", + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "index": "Zd", + "frameId": null, + "crop": null + }, + { + "type": "image", + "version": 991, + "versionNonce": 1413399916, + "isDeleted": false, + "id": "cCPdYqmyptSOVBzm7mwa0", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1607.6125069210364, + "y": 73.98942026818725, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 66.73359059732319, + "height": 26.316731328272667, + "seed": 535597875, + "groupIds": [ + "yKiV6gPTanKGtvKyuBZkk", + "hKl7RjmIcNyVk0DtQRD7k", + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "status": "saved", + "fileId": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "scale": [ + 1, + 1 + ], + "index": "Ze", + "frameId": null, + "crop": null + }, + { + "type": "text", + "version": 861, + "versionNonce": 1637562348, + "isDeleted": false, + "id": "1rN_kHhApPEKx2EMm4MbX", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1534.540347395779, + "y": 44.86844707706592, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 40.309179626377656, + "height": 10.785545626341246, + "seed": 1710863571, + "groupIds": [ + "hKl7RjmIcNyVk0DtQRD7k", + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "fontSize": 8.628436501072995, + "fontFamily": 1, + "text": "Tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Tenant A", + "lineHeight": 1.25, + "baseline": 8.000000000000002, + "index": "Zf", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 823, + "versionNonce": 533027436, + "isDeleted": false, + "id": "NhFzqMhqC7Wd9-fMvGCJE", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1805.7638979943972, + "y": 112.56256573616668, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 32.25597332253863, + "height": 8.628436501072997, + "seed": 138825011, + "groupIds": [ + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "fontSize": 6.902749200858397, + "fontFamily": 1, + "text": "Tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Tenant A", + "lineHeight": 1.25, + "baseline": 6.000000000000002, + "index": "Zu", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 1884, + "versionNonce": 25983188, + "isDeleted": false, + "id": "e-8ZsFtNd0YWIHs4OXK9I", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1788.6769864991456, + "y": 81.67581065587405, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 80.71732338696665, + "height": 4.706901614007698, + "seed": 1326198483, + "groupIds": [ + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044445997, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "I7fko55wggHsMjE_Zjhfo", + "focus": 0.16611802797687533, + "gap": 1.7433399242431733, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -80.71732338696665, + 4.706901614007698 + ] + ], + "index": "Zv", + "frameId": null + }, + { + "type": "text", + "version": 619, + "versionNonce": 812298092, + "isDeleted": false, + "id": "8L8CgMD_SJbwW4m3ZHH1h", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1539.382072347784, + "y": -14.66916710878644, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 17.803340537497707, + "height": 21.552711135428872, + "seed": 837082227, + "groupIds": [ + "7WNiCalXRW1si4FjlXFcH", + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "fontSize": 17.242168908343096, + "fontFamily": 1, + "text": "**", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "**", + "lineHeight": 1.25, + "baseline": 15, + "index": "Zw", + "frameId": null, + "autoResize": true + }, + { + "id": "3liIJkbvFXnj-rYEZQ8xJ", + "type": "rectangle", + "x": 1417.5202294808596, + "y": -38.52109535729301, + "width": 492.6786223434542, + "height": 243.97521029675846, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [ + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": { + "type": 3 + }, + "seed": 223709939, + "version": 624, + "versionNonce": 1689903596, + "isDeleted": false, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "index": "Zx", + "frameId": null + }, + { + "id": "WUFKEGLN-QDhP0qk762uj", + "type": "text", + "x": 1486.01217298042, + "y": -95.35744455682811, + "width": 52.56666564941406, + "height": 20, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [ + "LLydjUS263Z6iJ4YHblsT" + ], + "roundness": null, + "seed": 1240915923, + "version": 310, + "versionNonce": 1568588908, + "isDeleted": false, + "boundElements": [], + "updated": 1737044445859, + "link": null, + "locked": false, + "text": "ignored", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "ignored", + "lineHeight": 1.25, + "index": "Zz", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 881, + "versionNonce": 1497879532, + "isDeleted": false, + "id": "RaNUQl2ish66v6i9utUt1", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 279.7899715296179, + "y": 338.0219655990167, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 351.1963329295919, + "height": 178.41168954841385, + "seed": 1886192179, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "viZWH67LUjGqaj3n_tH0Q", + "type": "arrow" + }, + { + "id": "Osh8DpDAtYfWhIxQTpdhN", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "a0", + "frameId": null + }, + { + "type": "rectangle", + "version": 753, + "versionNonce": 515944660, + "isDeleted": false, + "id": "E0_cqgLTCZYdblXeBNRce", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 246.0357389853923, + "y": -118.75289187826343, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 953.4529315439731, + "height": 708.9963067219029, + "seed": 1175240413, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "a1", + "frameId": null + }, + { + "type": "text", + "version": 193, + "versionNonce": 137886676, + "isDeleted": false, + "id": "q_2M1iXS5kZHhdte9Lik-", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 287.5488650331077, + "y": -156.5896962253754, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 121.13990294933319, + "height": 25, + "seed": 1258711965, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044371326, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "K8s Cluster", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "K8s Cluster", + "lineHeight": 1.25, + "baseline": 18, + "index": "a3", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 241, + "versionNonce": 2075969772, + "isDeleted": false, + "id": "Pf29U31FXugJR3y16teDk", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 411.6691523387309, + "y": -87.01816200445023, + "strokeColor": "#000000", + "backgroundColor": "#a5d8ff", + "width": 229.49400491361894, + "height": 134.93853507796484, + "seed": 1104211965, + "groupIds": [ + "2xLoSSjPhMgeqTcoAJ45m" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "a4", + "frameId": null + }, + { + "type": "rectangle", + "version": 364, + "versionNonce": 1107397588, + "isDeleted": false, + "id": "W7YGziUK06CJHuE-QzOK2", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 415.11648717648916, + "y": -83.5708271666922, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 222.5993352381026, + "height": 44.32287648546298, + "seed": 428599389, + "groupIds": [ + "2xLoSSjPhMgeqTcoAJ45m" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "a5", + "frameId": null + }, + { + "type": "text", + "version": 313, + "versionNonce": 26004332, + "isDeleted": false, + "id": "TBkpSzAa8wHxnqT9E5CH2", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 433.3381141760684, + "y": -72.24386984262946, + "strokeColor": "#1e1e1e", + "backgroundColor": "#1971c2", + "width": 187.93333435058594, + "height": 21.507525856732585, + "seed": 1141429437, + "groupIds": [ + "2xLoSSjPhMgeqTcoAJ45m" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 17.20602068538607, + "fontFamily": 1, + "text": "Namespace A (tenant)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Namespace A (tenant)", + "lineHeight": 1.25, + "baseline": 15, + "index": "a6", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1647, + "versionNonce": 1124659540, + "isDeleted": false, + "id": "evHzKLVy0EOUfnX7mLgwn", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 465.62278026704223, + "y": -31.678821222182478, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 593366589, + "groupIds": [ + "d3WFboVCh_A2k7YVjEh9d", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ], + "index": "a7", + "frameId": null + }, + { + "type": "line", + "version": 986, + "versionNonce": 54033900, + "isDeleted": false, + "id": "q-_In0nWigaA4NFversey", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 464.9287401175453, + "y": -32.40583659657216, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 5979805, + "groupIds": [ + "d3WFboVCh_A2k7YVjEh9d", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ], + "index": "a8", + "frameId": null + }, + { + "type": "line", + "version": 683, + "versionNonce": 1225238228, + "isDeleted": false, + "id": "WF4Fm2n3yzQKdphnhCcfQ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 452.9320826343916, + "y": -9.685450544942455, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 11.606398134643495, + "height": 20.43580530455634, + "seed": 991126269, + "groupIds": [ + "IEPHLG7g-hTwmKjiyyTby", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.10680734479733643, + 14.205376858045264 + ], + [ + 11.321578548517282, + 20.43580530455634 + ], + [ + 11.606398134643495, + 3.8806668609697628 + ], + [ + 1.3528930340995515, + 1.032470999707551 + ] + ], + "index": "a9", + "frameId": null + }, + { + "type": "line", + "version": 901, + "versionNonce": 1774837868, + "isDeleted": false, + "id": "7fHaJR-Tl4E0RzwR8kL4r", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 478.3522306961569, + "y": -9.756655441473981, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 12.667312471753975, + "height": 20.509145060610187, + "seed": 1208482653, + "groupIds": [ + "IEPHLG7g-hTwmKjiyyTby", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -12.667312471753975, + 3.6435687166298947 + ], + [ + -11.887638165338293, + 20.509145060610187 + ], + [ + -0.5696391722524426, + 14.347786651108377 + ], + [ + 0, + 0 + ] + ], + "index": "aA", + "frameId": null + }, + { + "type": "line", + "version": 616, + "versionNonce": 1486072916, + "isDeleted": false, + "id": "FnQi7TijNDGbvWKKbar2z", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 453.8577462893019, + "y": -11.32316316516824, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 24.458881958589213, + "height": 7.5121165840790765, + "seed": 260736957, + "groupIds": [ + "IEPHLG7g-hTwmKjiyyTby", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.962422617301264, + 3.4890399300462054 + ], + [ + 24.458881958589213, + 0.28481958612622416 + ], + [ + 11.855615272503934, + -4.023076654032871 + ], + [ + 0, + 0 + ] + ], + "index": "aB", + "frameId": null + }, + { + "type": "rectangle", + "version": 1356, + "versionNonce": 1829434092, + "isDeleted": false, + "id": "GEPrKk2_r33AcFZt-4owF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 985.0696693817434, + "y": -51.79276395198133, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 2134847229, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + }, + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "aC", + "frameId": null + }, + { + "type": "rectangle", + "version": 1490, + "versionNonce": 1937965524, + "isDeleted": false, + "id": "KVhxDF9yv_MWK13J3dHL3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 987.4079346815705, + "y": -49.45449865215426, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 752502621, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "aD", + "frameId": null + }, + { + "type": "text", + "version": 1492, + "versionNonce": 704207212, + "isDeleted": false, + "id": "R1JboiRKt3oNY9o9xC077", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1011.8681011442326, + "y": -41.77162695272227, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 1009996733, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044569127, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aE", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 760, + "versionNonce": 491189204, + "isDeleted": false, + "id": "5cjKeARw4GNFEEJZ-9VN4", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1016.1405360867308, + "y": -5.529571655364862, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 99.51992797851562, + "height": 40, + "seed": 694932509, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": null, + "boundElements": [ + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + }, + { + "id": "4OxoKFs1_Xdbzn0FDWUjv", + "type": "arrow" + } + ], + "updated": 1737044579416, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Observability\nOperator*", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Observability\nOperator*", + "lineHeight": 1.25, + "baseline": 34, + "index": "aF", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 1100, + "versionNonce": 579086316, + "isDeleted": false, + "id": "maCvw5JncD4BKuAguGF0c", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 717.0696693817434, + "y": -94.19780849054655, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 141.52638459323293, + "seed": 1886734461, + "groupIds": [ + "BSJ9NpvgfUQydZ2CrVzeX", + "RUi1FtLYA-77GfM1YBh0Y" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "jqn7L1CPT_HyliDPI9P-j", + "type": "arrow" + }, + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "aG", + "frameId": null + }, + { + "type": "rectangle", + "version": 1198, + "versionNonce": 1428397268, + "isDeleted": false, + "id": "dvmQ_NYaRcNJZOZm3v1OF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 719.4079346815705, + "y": -91.85954319071948, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 319893725, + "groupIds": [ + "BSJ9NpvgfUQydZ2CrVzeX", + "RUi1FtLYA-77GfM1YBh0Y" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "aH", + "frameId": null + }, + { + "type": "text", + "version": 1105, + "versionNonce": 1342166868, + "isDeleted": false, + "id": "l7HSSaoTOrfjhrJnj8QIl", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 746.8681011442326, + "y": -86.1766714912875, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 1590758717, + "groupIds": [ + "RUi1FtLYA-77GfM1YBh0Y" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044565138, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aI", + "frameId": null, + "autoResize": true + }, + { + "type": "image", + "version": 652, + "versionNonce": 700506708, + "isDeleted": false, + "id": "YwqW6r2OmqYpQ8qE_dmeC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 756.686885052514, + "y": -42.179340890517096, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 1930845597, + "groupIds": [ + "6-seN9eTnZZzWPC82Jr-R", + "pWqSYLnDuN9C5Ko6KXKh1", + "OQDO0bZJrfwwUc2POj11y" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "index": "aJ", + "frameId": null, + "crop": null + }, + { + "type": "arrow", + "version": 444, + "versionNonce": 2029264108, + "isDeleted": false, + "id": "jqn7L1CPT_HyliDPI9P-j", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 715.9005000759886, + "y": -10.78325670726747, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 220, + "height": 17.1039158167504, + "seed": 897006461, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": { + "elementId": "maCvw5JncD4BKuAguGF0c", + "focus": -0.08474347907800747, + "gap": 1.1691693057548491 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -220, + 17.1039158167504 + ] + ], + "index": "aK", + "frameId": null + }, + { + "type": "text", + "version": 177, + "versionNonce": 1436121044, + "isDeleted": false, + "id": "r9D6VpvKhMXXNF5ZT-yj8", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 509.567168014221, + "y": -30.679340890517096, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 122.66666412353516, + "height": 20, + "seed": 1767687133, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "scrapes metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "scrapes metrics", + "lineHeight": 1.25, + "baseline": 14, + "index": "aL", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 159, + "versionNonce": 79273836, + "isDeleted": false, + "id": "TErMRO_X_1wfaHA1LWMjA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 751.350500838928, + "y": 20.320659109482904, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 103.0999984741211, + "height": 20, + "seed": 1861251901, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "prometheus-A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "prometheus-A", + "lineHeight": 1.25, + "baseline": 14, + "index": "aM", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 409, + "versionNonce": 1958960468, + "isDeleted": false, + "id": "QwDloiGA_JLdFTmuU7unS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 975.4699826731477, + "y": -3.436100817368242, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 98.5694825971591, + "height": 4.3436520113717165, + "seed": 1580191005, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": { + "elementId": "KdeBQ3YHl9MqzPsBoCQuZ", + "focus": -1.9242239307487354, + "gap": 13.795993107279287 + }, + "endBinding": { + "elementId": "maCvw5JncD4BKuAguGF0c", + "focus": 0.16229811829395357, + "gap": 4.169169305754167 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -98.5694825971591, + -4.3436520113717165 + ] + ], + "index": "aN", + "frameId": null + }, + { + "type": "text", + "version": 194, + "versionNonce": 833278444, + "isDeleted": false, + "id": "KdeBQ3YHl9MqzPsBoCQuZ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 902.8785583602222, + "y": -37.23209392464753, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 66.53333282470703, + "height": 20, + "seed": 1609230813, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Manages", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Manages", + "lineHeight": 1.25, + "baseline": 14, + "index": "aO", + "frameId": null, + "autoResize": true + }, + { + "id": "0CN7aawE0LBj0L0-wH0AS", + "type": "rectangle", + "x": 591.3789957096226, + "y": 172.31129390870842, + "width": 369.0947704295919, + "height": 94.57612411084251, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 3 + }, + "seed": 1308409811, + "version": 697, + "versionNonce": 1381265132, + "isDeleted": false, + "boundElements": [ + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + } + ], + "updated": 1737044555676, + "link": null, + "locked": false, + "index": "aP", + "frameId": null + }, + { + "id": "fWUQvR4Xdx0mfMQ7adaaJ", + "type": "text", + "x": 599.5758251795551, + "y": 188.38305356153091, + "width": 372.73434257507324, + "height": 64.44513875177073, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 181975091, + "version": 571, + "versionNonce": 1120864236, + "isDeleted": false, + "boundElements": [ + { + "id": "1YAkuXN0udd0mDd8-0X3l", + "type": "arrow" + } + ], + "updated": 1737044548300, + "link": null, + "locked": false, + "text": "apiVersion: v1\nkind: Namespace\nmetadata:\n// Ommitted for brevity\n annotations:\n metrics.monitoring.adevinta.com/remote-write=namespace-a/my-remote-write", + "fontSize": 9.33987518141605, + "fontFamily": 2, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 62, + "containerId": null, + "originalText": "apiVersion: v1\nkind: Namespace\nmetadata:\n// Ommitted for brevity\n annotations:\n metrics.monitoring.adevinta.com/remote-write=namespace-a/my-remote-write", + "lineHeight": 1.15, + "index": "aQ", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1584, + "versionNonce": 396564564, + "isDeleted": false, + "id": "zUhv0FUCvdgCgSAyodVPH", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 594.7059752774054, + "y": 113.1715891846303, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 67.37474875735316, + "height": 67.47500880014684, + "seed": 220998355, + "groupIds": [ + "21zmTFQ4XmHnDWJaDcdr-", + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -26.869691468706304, + 12.933545520384756 + ], + [ + -33.38659425029553, + 43.111818401282534 + ], + [ + -14.93874637625835, + 67.07396862897212 + ], + [ + 14.938746376258385, + 67.47500880014684 + ], + [ + 33.98815450705763, + 43.011558358488855 + ], + [ + 27.37099168267472, + 12.83328547759108 + ], + [ + 0, + 0 + ] + ], + "index": "aR", + "frameId": null + }, + { + "type": "line", + "version": 923, + "versionNonce": 2046314220, + "isDeleted": false, + "id": "Q1fmEtTedxlYJAAmyhdyr", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 593.9344283311495, + "y": 112.36338451174788, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 67.37474875735316, + "height": 67.47500880014684, + "seed": 581261427, + "groupIds": [ + "21zmTFQ4XmHnDWJaDcdr-", + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -26.869691468706304, + 12.933545520384756 + ], + [ + -33.38659425029553, + 43.111818401282534 + ], + [ + -14.93874637625835, + 67.07396862897212 + ], + [ + 14.938746376258385, + 67.47500880014684 + ], + [ + 33.98815450705763, + 43.011558358488855 + ], + [ + 27.37099168267472, + 12.83328547759108 + ], + [ + 0, + 0 + ] + ], + "index": "aS", + "frameId": null + }, + { + "type": "rectangle", + "version": 489, + "versionNonce": 1975840212, + "isDeleted": false, + "id": "KH4s95NUqWRw2hUYctDtG", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "dotted", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 578.6408280564926, + "y": 132.02151130920402, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 30.994042307547033, + "height": 27.621332262823326, + "seed": 937939475, + "groupIds": [ + "21zmTFQ4XmHnDWJaDcdr-", + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "aT", + "frameId": null + }, + { + "type": "text", + "version": 426, + "versionNonce": 1739098476, + "isDeleted": false, + "id": "CzfYIVdX3lo7Vn9MaQchT", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 587.2840316258653, + "y": 159.44371981189613, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 11.766666412353516, + "height": 14.593554970962579, + "seed": 501880755, + "groupIds": [ + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 11.674843976770063, + "fontFamily": 1, + "text": "ns", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "ns", + "lineHeight": 1.25, + "baseline": 10, + "index": "aU", + "frameId": null, + "autoResize": true + }, + { + "id": "nYzmlP6fDWXWoYoi0xwJC", + "type": "text", + "x": 637.8609346119969, + "y": 142.64035301118827, + "width": 57.36666488647461, + "height": 11.674843976770063, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [ + "3q-izW3IfC3KM17_DJnbz" + ], + "roundness": null, + "seed": 1401292979, + "version": 247, + "versionNonce": 739251028, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "text": "namespace-a", + "fontSize": 9.339875181416051, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 8, + "containerId": null, + "originalText": "namespace-a", + "lineHeight": 1.25, + "index": "aV", + "frameId": null, + "autoResize": true + }, + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow", + "x": 1055.9827315727512, + "y": 45.047506721984206, + "width": 119.52640706324314, + "height": 125.96360016488589, + "angle": 0, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 1568120189, + "version": 635, + "versionNonce": 1633769812, + "isDeleted": false, + "boundElements": [], + "updated": 1737044579417, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + -119.52640706324314, + 125.96360016488589 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "5cjKeARw4GNFEEJZ-9VN4", + "gap": 10.577078377349068, + "focus": -0.27781927268327866 + }, + "endBinding": { + "elementId": "0CN7aawE0LBj0L0-wH0AS", + "gap": 1.300187021838326, + "focus": 0.49372043709200675 + }, + "startArrowhead": null, + "endArrowhead": "triangle", + "index": "aW", + "frameId": null + }, + { + "type": "text", + "version": 435, + "versionNonce": 148539604, + "isDeleted": false, + "id": "Nl0-u8ODET-dMN_wpF6ze", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.562155244553513, + "x": 930.8789926297804, + "y": 104.28183345705474, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 183.23333740234375, + "height": 20, + "seed": 1431953427, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Reads NS configuration", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Reads NS configuration", + "lineHeight": 1.25, + "baseline": 14, + "index": "aX", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 2933, + "versionNonce": 2084434004, + "isDeleted": false, + "id": "wPDdf5cDNOM2csqXMlzht", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 279.8296426262556, + "y": 298.39318823673824, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 55.94758223976345, + "height": 56.33136628063863, + "seed": 1632487837, + "groupIds": [ + "hlW-3eoj3XLkl960YFqOv", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -21.82773103219606, + 10.540948194771797 + ], + [ + -26.917459040644793, + 35.51549993198945 + ], + [ + -11.162486299279974, + 55.17308761827916 + ], + [ + 13.346784725451252, + 55.74008239953474 + ], + [ + 29.030123199118655, + 35.59937404387765 + ], + [ + 24.97922081885728, + 12.693933788023726 + ], + [ + 0.4636712753502693, + -0.591283881103893 + ] + ], + "index": "ax", + "frameId": null + }, + { + "type": "line", + "version": 1779, + "versionNonce": 1662447340, + "isDeleted": false, + "id": "iiFSfbcrcJWFi0mUHOK1A", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 280.4362021855528, + "y": 297.9158416424688, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 55.46949186476053, + "height": 55.55203575146405, + "seed": 326929917, + "groupIds": [ + "hlW-3eoj3XLkl960YFqOv", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -22.12176163654139, + 10.648161384753138 + ], + [ + -27.48711427226972, + 35.49387128251047 + ], + [ + -12.29903911882338, + 55.22186020465 + ], + [ + 12.299039118823407, + 55.55203575146405 + ], + [ + 27.98237759249081, + 35.41132739580696 + ], + [ + 22.534481070058963, + 10.565617498049628 + ], + [ + 0, + 0 + ] + ], + "index": "ay", + "frameId": null + }, + { + "type": "line", + "version": 815, + "versionNonce": 826590676, + "isDeleted": false, + "id": "fh_qC7d7KFSa75crN1vqx", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 276.2910313104784, + "y": 314.85709451897577, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 18.04713752652419, + "height": 0.2765844831651, + "seed": 806602333, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.04713752652419, + -0.2765844831651 + ] + ], + "index": "az", + "frameId": null + }, + { + "type": "line", + "version": 881, + "versionNonce": 112341356, + "isDeleted": false, + "id": "39uHZyPXQvfGdPVHYgR06", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 276.6051839289627, + "y": 336.0039697178525, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 18.04713752652419, + "height": 0.2765844831651, + "seed": 971305661, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.04713752652419, + -0.2765844831651 + ] + ], + "index": "b00", + "frameId": null + }, + { + "type": "line", + "version": 925, + "versionNonce": 433384276, + "isDeleted": false, + "id": "QhhWFmWlJhp_Jf-0AEIFk", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 282.3443119546388, + "y": 329.1585037595158, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 11.6165482929351, + "height": 0.4840228455389318, + "seed": 1569821469, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.6165482929351, + -0.4840228455389318 + ] + ], + "index": "b01", + "frameId": null + }, + { + "type": "line", + "version": 997, + "versionNonce": 1254171628, + "isDeleted": false, + "id": "5kfBQlePKEBK3qykEbPpj", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 282.4307336584533, + "y": 322.6206851627639, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 11.6165482929351, + "height": 0.4840228455389318, + "seed": 931866493, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.6165482929351, + -0.4840228455389318 + ] + ], + "index": "b02", + "frameId": null + }, + { + "type": "line", + "version": 1036, + "versionNonce": 407134420, + "isDeleted": false, + "id": "MisOE3UownHjJr-yDzMVZ", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 267.08029484278916, + "y": 314.5305890301841, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 1511206877, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b03", + "frameId": null + }, + { + "type": "line", + "version": 1111, + "versionNonce": 1162724972, + "isDeleted": false, + "id": "5SRC_uisJvGi1Noe_7LnQ", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 274.6658250410927, + "y": 321.56103180185966, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 322645053, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b04", + "frameId": null + }, + { + "type": "line", + "version": 1092, + "versionNonce": 1411982932, + "isDeleted": false, + "id": "leH9PvGxWEaKVx1Z9aqQ5", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 274.59667892030154, + "y": 328.95966672652656, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 1927584925, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b05", + "frameId": null + }, + { + "type": "line", + "version": 1092, + "versionNonce": 992090348, + "isDeleted": false, + "id": "fPGvb0LlEaiERuJdUUxa5", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 268.02779744513, + "y": 336.15086328881983, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 281084157, + "groupIds": [ + "hrjFZHwgGWFqP8TshV5Gp", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b06", + "frameId": null + }, + { + "type": "ellipse", + "version": 1023, + "versionNonce": 307837908, + "isDeleted": false, + "id": "8DjsCqpL9LsdAKMW7jUXU", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 273.5964401112643, + "y": 315.9540096374721, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 15.206288818118333, + "height": 12.371536211283903, + "seed": 307353949, + "groupIds": [ + "0q8RHRqM8w9wYFDwyhLkh", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b07", + "frameId": null + }, + { + "type": "ellipse", + "version": 1220, + "versionNonce": 1204296556, + "isDeleted": false, + "id": "gKj4752g8a85sBUr9Dzzi", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 275.1921617773629, + "y": 318.7042828619836, + "strokeColor": "#228be6", + "backgroundColor": "#228be6", + "width": 11.97729909377715, + "height": 10.644402172682803, + "seed": 1894294973, + "groupIds": [ + "0q8RHRqM8w9wYFDwyhLkh", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b08", + "frameId": null + }, + { + "type": "rectangle", + "version": 1472, + "versionNonce": 216703316, + "isDeleted": false, + "id": "h2RqOL2LwlRA4l21_ZFT0", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 272.43604885760544, + "y": 323.36483535881456, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 17.461735710711398, + "height": 11.481902904411756, + "seed": 2141644317, + "groupIds": [ + "IstgHhlpPudgDc1VK3lk7", + "0q8RHRqM8w9wYFDwyhLkh", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b09", + "frameId": null + }, + { + "type": "ellipse", + "version": 717, + "versionNonce": 92917228, + "isDeleted": false, + "id": "PNNt0sQyRGnUNFNuTC6x2", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 279.42612250728814, + "y": 327.18091452666494, + "strokeColor": "#228be6", + "backgroundColor": "#228be6", + "width": 3.416104830716019, + "height": 2.881410161560464, + "seed": 830347901, + "groupIds": [ + "0q8RHRqM8w9wYFDwyhLkh", + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0A", + "frameId": null + }, + { + "type": "text", + "version": 707, + "versionNonce": 1261655764, + "isDeleted": false, + "id": "iZ9NFmi7xhrWvsV06IzxT", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 266.86205559132475, + "y": 337.4460919837516, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 30.123583386664713, + "height": 12.014843745913002, + "seed": 523223773, + "groupIds": [ + "9SaDiqCMNlu-omJmRZG_l" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.611874996730402, + "fontFamily": 1, + "text": "secret", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "secret", + "lineHeight": 1.25, + "baseline": 9, + "index": "b0B", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 346, + "versionNonce": 896610412, + "isDeleted": false, + "id": "H3YuU5XWJ0qwkN-Yva3it", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 322.5454666741945, + "y": 312.78522406798487, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 70.56666564941406, + "height": 11.674843976770063, + "seed": 1337781619, + "groupIds": [ + "gVjRCNBBOcXQTGcY7ywfZ" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.339875181416051, + "fontFamily": 1, + "text": "my-remote-write", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "my-remote-write", + "lineHeight": 1.25, + "baseline": 8, + "index": "b0C", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1785, + "versionNonce": 2043239508, + "isDeleted": false, + "id": "V9bkpJiIci8vHbAVRrEtc", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 310.9287607351263, + "y": 287.42034767237476, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 21.024773235700497, + "height": 21.05606010063458, + "seed": 1675172243, + "groupIds": [ + "vn6dKD-RPwdFWirAx4hVa", + "WV9kXrazICUi8pq2thi7F", + "U-I7ZQqLxqPGNW8sK_rvh" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -8.384879802332932, + 4.036005576496078 + ], + [ + -10.418526023048011, + 13.453351921653596 + ], + [ + -4.6617428751776355, + 20.930912640898267 + ], + [ + 4.661742875177646, + 21.05606010063458 + ], + [ + 10.606247212652486, + 13.422065056719518 + ], + [ + 8.541314127003327, + 4.004718711562001 + ], + [ + 0, + 0 + ] + ], + "index": "b0D", + "frameId": null + }, + { + "type": "line", + "version": 1124, + "versionNonce": 1732788972, + "isDeleted": false, + "id": "vhqzZQ_aN_MdFw9cDbgt_", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 310.6879939807426, + "y": 287.1681416116652, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 21.024773235700497, + "height": 21.05606010063458, + "seed": 1531342643, + "groupIds": [ + "vn6dKD-RPwdFWirAx4hVa", + "WV9kXrazICUi8pq2thi7F", + "U-I7ZQqLxqPGNW8sK_rvh" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -8.384879802332932, + 4.036005576496078 + ], + [ + -10.418526023048011, + 13.453351921653596 + ], + [ + -4.6617428751776355, + 20.930912640898267 + ], + [ + 4.661742875177646, + 21.05606010063458 + ], + [ + 10.606247212652486, + 13.422065056719518 + ], + [ + 8.541314127003327, + 4.004718711562001 + ], + [ + 0, + 0 + ] + ], + "index": "b0E", + "frameId": null + }, + { + "type": "rectangle", + "version": 689, + "versionNonce": 1115861460, + "isDeleted": false, + "id": "wZtFf0WPlsRHoKplYcVMV", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "dotted", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 305.91551640327305, + "y": 293.3026009718577, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 9.671913041497833, + "height": 8.619434699271046, + "seed": 1371017427, + "groupIds": [ + "vn6dKD-RPwdFWirAx4hVa", + "WV9kXrazICUi8pq2thi7F", + "U-I7ZQqLxqPGNW8sK_rvh" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0F", + "frameId": null + }, + { + "type": "text", + "version": 626, + "versionNonce": 563676524, + "isDeleted": false, + "id": "Cb8jrYCzxmViTaA-i9VWU", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 308.6126900243654, + "y": 301.8598976746269, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 3.6718725876193545, + "height": 4.554023423111193, + "seed": 2026143347, + "groupIds": [ + "WV9kXrazICUi8pq2thi7F", + "U-I7ZQqLxqPGNW8sK_rvh" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 3.6432187384889545, + "fontFamily": 1, + "text": "ns", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "ns", + "lineHeight": 1.25, + "baseline": 3, + "index": "b0G", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 446, + "versionNonce": 611669844, + "isDeleted": false, + "id": "Q9O5RNwkByjg3XDqVCCEO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 326.96455062928123, + "y": 296.12566911219096, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 33.18778101144198, + "height": 6.754134409078216, + "seed": 187840531, + "groupIds": [ + "U-I7ZQqLxqPGNW8sK_rvh" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 5.403307527262574, + "fontFamily": 1, + "text": "namespace-a", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "namespace-a", + "lineHeight": 1.25, + "baseline": 4.999999999999999, + "index": "b0H", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 756, + "versionNonce": 1548026092, + "isDeleted": false, + "id": "yWwz62tI1-_nmrJMxkBgC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 297.66550506020224, + "y": 354.35769411851635, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 325.62438774108887, + "height": 161.11284687942683, + "seed": 1594058227, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "1YAkuXN0udd0mDd8-0X3l", + "type": "arrow" + } + ], + "updated": 1737044523121, + "link": null, + "locked": false, + "fontSize": 9.33987518141605, + "fontFamily": 2, + "text": "apiVersion: v1\nkind: Secret\nmetadata:\n annotations:\n monitor.adevinta.com/referenced-secrets: my-username,my-password\nstringData:\n remote-write: |\n url: https://my-remote-write\n basicAuth:\n username:\n name: my-rw-username\n key: user-name\n password:\n name: my-rw-password\n key: password", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "apiVersion: v1\nkind: Secret\nmetadata:\n annotations:\n monitor.adevinta.com/referenced-secrets: my-username,my-password\nstringData:\n remote-write: |\n url: https://my-remote-write\n basicAuth:\n username:\n name: my-rw-username\n key: user-name\n password:\n name: my-rw-password\n key: password", + "lineHeight": 1.15, + "baseline": 158, + "index": "b0I", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 1066, + "versionNonce": 1274498260, + "isDeleted": false, + "id": "oi9GU0q2xYWaUaLFTOKgl", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 775.6764303257451, + "y": 332.6592616450766, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 351.1963329295919, + "height": 94.57612411084251, + "seed": 809710845, + "groupIds": [ + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "viZWH67LUjGqaj3n_tH0Q", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0J", + "frameId": null + }, + { + "type": "line", + "version": 3178, + "versionNonce": 193960556, + "isDeleted": false, + "id": "SzHROSVcFpm_QOftRg9u0", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 775.7161014223827, + "y": 293.030484282798, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 55.94758223976345, + "height": 56.33136628063863, + "seed": 1802341725, + "groupIds": [ + "Bk9mgEz3-uQOqxZmIaGQN", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -21.82773103219606, + 10.540948194771797 + ], + [ + -26.917459040644793, + 35.51549993198945 + ], + [ + -11.162486299279974, + 55.17308761827916 + ], + [ + 13.346784725451252, + 55.74008239953474 + ], + [ + 29.030123199118655, + 35.59937404387765 + ], + [ + 24.97922081885728, + 12.693933788023726 + ], + [ + 0.4636712753502693, + -0.591283881103893 + ] + ], + "index": "b0K", + "frameId": null + }, + { + "type": "line", + "version": 2024, + "versionNonce": 1788777044, + "isDeleted": false, + "id": "dOTXSxrhbMnNO5wh4tE5R", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 776.32266098168, + "y": 292.5531376885286, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 55.46949186476053, + "height": 55.55203575146405, + "seed": 342743485, + "groupIds": [ + "Bk9mgEz3-uQOqxZmIaGQN", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -22.12176163654139, + 10.648161384753138 + ], + [ + -27.48711427226972, + 35.49387128251047 + ], + [ + -12.29903911882338, + 55.22186020465 + ], + [ + 12.299039118823407, + 55.55203575146405 + ], + [ + 27.98237759249081, + 35.41132739580696 + ], + [ + 22.534481070058963, + 10.565617498049628 + ], + [ + 0, + 0 + ] + ], + "index": "b0L", + "frameId": null + }, + { + "type": "line", + "version": 1060, + "versionNonce": 34172140, + "isDeleted": false, + "id": "Dxb1_CXLrAR0BeOpWTNK0", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 772.1774901066055, + "y": 309.49439056503553, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 18.04713752652419, + "height": 0.2765844831651, + "seed": 1684380189, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.04713752652419, + -0.2765844831651 + ] + ], + "index": "b0M", + "frameId": null + }, + { + "type": "line", + "version": 1126, + "versionNonce": 255928276, + "isDeleted": false, + "id": "mU0HhxrAkBX9WrOeZbF3I", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 772.4916427250898, + "y": 330.64126576391214, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 18.04713752652419, + "height": 0.2765844831651, + "seed": 1950764669, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.04713752652419, + -0.2765844831651 + ] + ], + "index": "b0N", + "frameId": null + }, + { + "type": "line", + "version": 1170, + "versionNonce": 1587867500, + "isDeleted": false, + "id": "h2A8NdrlZLVfGtFZeCmyv", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 778.2307707507659, + "y": 323.7957998055754, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 11.6165482929351, + "height": 0.4840228455389318, + "seed": 1121023709, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.6165482929351, + -0.4840228455389318 + ] + ], + "index": "b0O", + "frameId": null + }, + { + "type": "line", + "version": 1242, + "versionNonce": 2113211732, + "isDeleted": false, + "id": "7HjjeBQxCvbOC8fkri9xG", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 778.3171924545804, + "y": 317.25798120882365, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 11.6165482929351, + "height": 0.4840228455389318, + "seed": 2056819517, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.6165482929351, + -0.4840228455389318 + ] + ], + "index": "b0P", + "frameId": null + }, + { + "type": "line", + "version": 1284, + "versionNonce": 1370235372, + "isDeleted": false, + "id": "C0INdlfCrWSCkBT4-FFG-", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 762.9667536389163, + "y": 309.16788507624375, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 2013078429, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0Q", + "frameId": null + }, + { + "type": "line", + "version": 1356, + "versionNonce": 135283412, + "isDeleted": false, + "id": "n7jQM54c1WAAOYjG9qMP4", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 770.5522838372199, + "y": 316.1983278479193, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 254846973, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0R", + "frameId": null + }, + { + "type": "line", + "version": 1337, + "versionNonce": 1915290732, + "isDeleted": false, + "id": "hZi4ZnTR5FQwb1tAGhJ5c", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 770.4831377164287, + "y": 323.5969627725863, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 305438813, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0S", + "frameId": null + }, + { + "type": "line", + "version": 1340, + "versionNonce": 1680394324, + "isDeleted": false, + "id": "dCMbL4-sAw_D1c9-ySGUN", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 763.9142562412571, + "y": 330.7881593348796, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 383421629, + "groupIds": [ + "Y3LsIiFQU6biKta89P7Jd", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0T", + "frameId": null + }, + { + "type": "ellipse", + "version": 1268, + "versionNonce": 1709545196, + "isDeleted": false, + "id": "Oyie-geF3cO1MwJ86bqYZ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 769.4828989073916, + "y": 310.59130568353186, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 15.206288818118333, + "height": 12.371536211283903, + "seed": 2093215005, + "groupIds": [ + "e1jl8I1s5IJlOfEqOeFdu", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0U", + "frameId": null + }, + { + "type": "ellipse", + "version": 1465, + "versionNonce": 102014420, + "isDeleted": false, + "id": "aPdU3278z-5a58xasN1eF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 771.0786205734901, + "y": 313.34157890804323, + "strokeColor": "#228be6", + "backgroundColor": "#228be6", + "width": 11.97729909377715, + "height": 10.644402172682803, + "seed": 862077309, + "groupIds": [ + "e1jl8I1s5IJlOfEqOeFdu", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0V", + "frameId": null + }, + { + "type": "rectangle", + "version": 1717, + "versionNonce": 158062956, + "isDeleted": false, + "id": "HYv4Q7fUr79ZAaj0_Nj35", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 768.3225076537327, + "y": 318.0021314048743, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 17.461735710711398, + "height": 11.481902904411756, + "seed": 1073154525, + "groupIds": [ + "unUwskmnQAABeG8q6x6dC", + "e1jl8I1s5IJlOfEqOeFdu", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0W", + "frameId": null + }, + { + "type": "ellipse", + "version": 962, + "versionNonce": 1995031380, + "isDeleted": false, + "id": "M9FQhVg_yIH719k7GfvMW", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 775.3125813034153, + "y": 321.8182105727247, + "strokeColor": "#228be6", + "backgroundColor": "#228be6", + "width": 3.416104830716019, + "height": 2.881410161560464, + "seed": 60016189, + "groupIds": [ + "e1jl8I1s5IJlOfEqOeFdu", + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0X", + "frameId": null + }, + { + "type": "text", + "version": 963, + "versionNonce": 403033068, + "isDeleted": false, + "id": "oENOXN4ZThjXmSRovlNhe", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 762.7485143874521, + "y": 332.08338802981126, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 30.123583386664713, + "height": 12.014843745913002, + "seed": 1092220573, + "groupIds": [ + "i2c-dIsBT7xATL_0E2Ylg", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.611874996730402, + "fontFamily": 1, + "text": "secret", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "secret", + "lineHeight": 1.25, + "baseline": 9, + "index": "b0Y", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 626, + "versionNonce": 754858196, + "isDeleted": false, + "id": "a4YSYeFN2GqWEpIF-eGjO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 817.1616896303584, + "y": 306.7874021940631, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 68.69999694824219, + "height": 11.674843976770063, + "seed": 1664817917, + "groupIds": [ + "7qz8Dl-8i19qzRsaY4Zjc", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.339875181416051, + "fontFamily": 1, + "text": "my-rw-username", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "my-rw-username", + "lineHeight": 1.25, + "baseline": 8, + "index": "b0Z", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 2052, + "versionNonce": 1404923500, + "isDeleted": false, + "id": "D2u39TmKQOf7YHE84deEL", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 805.5449836912902, + "y": 281.4225257984528, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 21.024773235700497, + "height": 21.05606010063458, + "seed": 41464669, + "groupIds": [ + "JW1VoGoINYCDz8kgMjAkZ", + "UwO2TWwsfbumNOdyYDt3m", + "cuKEyYmgzpEgmxLHgEUy5", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -8.384879802332932, + 4.036005576496078 + ], + [ + -10.418526023048011, + 13.453351921653596 + ], + [ + -4.6617428751776355, + 20.930912640898267 + ], + [ + 4.661742875177646, + 21.05606010063458 + ], + [ + 10.606247212652486, + 13.422065056719518 + ], + [ + 8.541314127003327, + 4.004718711562001 + ], + [ + 0, + 0 + ] + ], + "index": "b0a", + "frameId": null + }, + { + "type": "line", + "version": 1391, + "versionNonce": 1676148308, + "isDeleted": false, + "id": "lwwcunnbVdBjk_rHDziJU", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 805.3042169369063, + "y": 281.1703197377432, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 21.024773235700497, + "height": 21.05606010063458, + "seed": 723795901, + "groupIds": [ + "JW1VoGoINYCDz8kgMjAkZ", + "UwO2TWwsfbumNOdyYDt3m", + "cuKEyYmgzpEgmxLHgEUy5", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -8.384879802332932, + 4.036005576496078 + ], + [ + -10.418526023048011, + 13.453351921653596 + ], + [ + -4.6617428751776355, + 20.930912640898267 + ], + [ + 4.661742875177646, + 21.05606010063458 + ], + [ + 10.606247212652486, + 13.422065056719518 + ], + [ + 8.541314127003327, + 4.004718711562001 + ], + [ + 0, + 0 + ] + ], + "index": "b0b", + "frameId": null + }, + { + "type": "rectangle", + "version": 957, + "versionNonce": 761053420, + "isDeleted": false, + "id": "3v5X6R93NgEjbZ_F2-ga3", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "dotted", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 801.1668572794187, + "y": 287.30477909793575, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 9.671913041497833, + "height": 8.619434699271046, + "seed": 1667186717, + "groupIds": [ + "JW1VoGoINYCDz8kgMjAkZ", + "UwO2TWwsfbumNOdyYDt3m", + "cuKEyYmgzpEgmxLHgEUy5", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0c", + "frameId": null + }, + { + "type": "text", + "version": 893, + "versionNonce": 1954354132, + "isDeleted": false, + "id": "CBYszkbZSIC3YCJRc7GNo", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 803.2289129805293, + "y": 295.862075800705, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 3.6718725876193545, + "height": 4.554023423111193, + "seed": 1897608317, + "groupIds": [ + "UwO2TWwsfbumNOdyYDt3m", + "cuKEyYmgzpEgmxLHgEUy5", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 3.6432187384889545, + "fontFamily": 1, + "text": "ns", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "ns", + "lineHeight": 1.25, + "baseline": 3, + "index": "b0d", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 713, + "versionNonce": 1819132780, + "isDeleted": false, + "id": "AmY89yY1cjfr7arO-Qh6P", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 821.580773585445, + "y": 290.1278472382691, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 33.18778101144198, + "height": 6.754134409078216, + "seed": 1612291293, + "groupIds": [ + "cuKEyYmgzpEgmxLHgEUy5", + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 5.403307527262574, + "fontFamily": 1, + "text": "namespace-a", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "namespace-a", + "lineHeight": 1.25, + "baseline": 5, + "index": "b0e", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 968, + "versionNonce": 752056660, + "isDeleted": false, + "id": "f0cXf1zYqXLHzdoL--0hq", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 808.120750720912, + "y": 343.6317725994307, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 103.76667022705078, + "height": 64.44513875177073, + "seed": 1045908243, + "groupIds": [ + "KqBYFuv333D2owqfI0q3f" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.33987518141605, + "fontFamily": 2, + "text": "apiVersion: v1\nkind: Secret\nmetadata:\n name: my-rw-username\nstringData:\n username: admin", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "apiVersion: v1\nkind: Secret\nmetadata:\n name: my-rw-username\nstringData:\n username: admin", + "lineHeight": 1.15, + "baseline": 62, + "index": "b0f", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 1145, + "versionNonce": 1463449068, + "isDeleted": false, + "id": "tpVrLynf34UbAU8BamkXR", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 773.4910772888527, + "y": 469.5693453241487, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 351.1963329295919, + "height": 94.57612411084251, + "seed": 1337236221, + "groupIds": [ + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0g", + "frameId": null + }, + { + "type": "line", + "version": 3225, + "versionNonce": 675360468, + "isDeleted": false, + "id": "viOZmgbasCaAfh8xQKDpd", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 773.5307483854904, + "y": 429.30545004188855, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 55.94758223976345, + "height": 56.33136628063863, + "seed": 599929693, + "groupIds": [ + "V9k7LPmvDvCAJbbatCH1_", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -21.82773103219606, + 10.540948194771797 + ], + [ + -26.917459040644793, + 35.51549993198945 + ], + [ + -11.162486299279974, + 55.17308761827916 + ], + [ + 13.346784725451252, + 55.74008239953474 + ], + [ + 29.030123199118655, + 35.59937404387765 + ], + [ + 24.97922081885728, + 12.693933788023726 + ], + [ + 0.4636712753502693, + -0.591283881103893 + ] + ], + "index": "b0h", + "frameId": null + }, + { + "type": "line", + "version": 2103, + "versionNonce": 1523374188, + "isDeleted": false, + "id": "7j4yCouTozfg8ut1QhyEz", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 774.1373079447876, + "y": 428.8281034476191, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 55.46949186476053, + "height": 55.55203575146405, + "seed": 1732100029, + "groupIds": [ + "V9k7LPmvDvCAJbbatCH1_", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -22.12176163654139, + 10.648161384753138 + ], + [ + -27.48711427226972, + 35.49387128251047 + ], + [ + -12.29903911882338, + 55.22186020465 + ], + [ + 12.299039118823407, + 55.55203575146405 + ], + [ + 27.98237759249081, + 35.41132739580696 + ], + [ + 22.534481070058963, + 10.565617498049628 + ], + [ + 0, + 0 + ] + ], + "index": "b0i", + "frameId": null + }, + { + "type": "line", + "version": 1107, + "versionNonce": 660354132, + "isDeleted": false, + "id": "JRjcUeypsGGHzsMnTOD1X", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 769.9921370697132, + "y": 445.7693563241261, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 18.04713752652419, + "height": 0.2765844831651, + "seed": 1615380509, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.04713752652419, + -0.2765844831651 + ] + ], + "index": "b0j", + "frameId": null + }, + { + "type": "line", + "version": 1173, + "versionNonce": 854518508, + "isDeleted": false, + "id": "eNe0EHWHDZ42GR9JCftmy", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 770.3062896881975, + "y": 466.9162315230027, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 18.04713752652419, + "height": 0.2765844831651, + "seed": 349484157, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.04713752652419, + -0.2765844831651 + ] + ], + "index": "b0k", + "frameId": null + }, + { + "type": "line", + "version": 1217, + "versionNonce": 1744472532, + "isDeleted": false, + "id": "5_orLTp3nhxsYOf4vfuo6", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 776.0454177138736, + "y": 460.070765564666, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 11.6165482929351, + "height": 0.4840228455389318, + "seed": 1352867037, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.6165482929351, + -0.4840228455389318 + ] + ], + "index": "b0l", + "frameId": null + }, + { + "type": "line", + "version": 1289, + "versionNonce": 304786796, + "isDeleted": false, + "id": "W9OL4ZDk215kcW7LyK_-r", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 776.1318394176881, + "y": 453.5329469679142, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 11.6165482929351, + "height": 0.4840228455389318, + "seed": 1799715133, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.6165482929351, + -0.4840228455389318 + ] + ], + "index": "b0m", + "frameId": null + }, + { + "type": "line", + "version": 1329, + "versionNonce": 2100440916, + "isDeleted": false, + "id": "LK4-G4Idn576I7xV09obH", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 760.781400602024, + "y": 445.4428508353343, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 1310542237, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0n", + "frameId": null + }, + { + "type": "line", + "version": 1403, + "versionNonce": 1237983212, + "isDeleted": false, + "id": "QVN9l5WQ7ujOIf0KhPOaZ", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 768.3669308003275, + "y": 452.47329360700985, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 979111421, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0o", + "frameId": null + }, + { + "type": "line", + "version": 1384, + "versionNonce": 660153556, + "isDeleted": false, + "id": "JCDvzOvLFdbtnAuNM68JH", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 768.2977846795363, + "y": 459.8719285316769, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 1325614685, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0p", + "frameId": null + }, + { + "type": "line", + "version": 1385, + "versionNonce": 1909932652, + "isDeleted": false, + "id": "AQFW70Q8rJ9fQHruV_fZo", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 761.7289032043648, + "y": 467.06312509397014, + "strokeColor": "#fff", + "backgroundColor": "#fff", + "width": 3.9413288851029686, + "height": 0.3457306039564228, + "seed": 1323594429, + "groupIds": [ + "bjw5dxc5shtwtqw5BlzfK", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9413288851029686, + 0.3457306039564228 + ] + ], + "index": "b0q", + "frameId": null + }, + { + "type": "ellipse", + "version": 1315, + "versionNonce": 738552404, + "isDeleted": false, + "id": "xeZCamStPvriOosyYL8uu", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 767.2975458704991, + "y": 446.8662714426224, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 15.206288818118333, + "height": 12.371536211283903, + "seed": 648090397, + "groupIds": [ + "6E4r8tNi8F7xt3iBcW43_", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0r", + "frameId": null + }, + { + "type": "ellipse", + "version": 1512, + "versionNonce": 287053036, + "isDeleted": false, + "id": "B5Duup4c5MuGGRTXbAyG2", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 768.8932675365977, + "y": 449.6165446671338, + "strokeColor": "#228be6", + "backgroundColor": "#228be6", + "width": 11.97729909377715, + "height": 10.644402172682803, + "seed": 850044797, + "groupIds": [ + "6E4r8tNi8F7xt3iBcW43_", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0s", + "frameId": null + }, + { + "type": "rectangle", + "version": 1764, + "versionNonce": 772603860, + "isDeleted": false, + "id": "vIppQLytoctB9THkgmkNi", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 766.1371546168402, + "y": 454.27709716396487, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 17.461735710711398, + "height": 11.481902904411756, + "seed": 1161998301, + "groupIds": [ + "Od64I08zC2fmv6U3CEpm_", + "6E4r8tNi8F7xt3iBcW43_", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0t", + "frameId": null + }, + { + "type": "ellipse", + "version": 1009, + "versionNonce": 1159894892, + "isDeleted": false, + "id": "MEK9NolbgyJGZy7SmiqOw", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 773.1272282665229, + "y": 458.09317633181524, + "strokeColor": "#228be6", + "backgroundColor": "#228be6", + "width": 3.416104830716019, + "height": 2.881410161560464, + "seed": 1819978813, + "groupIds": [ + "6E4r8tNi8F7xt3iBcW43_", + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0u", + "frameId": null + }, + { + "type": "text", + "version": 1000, + "versionNonce": 92661076, + "isDeleted": false, + "id": "2xmyGlYoJ9y8Bb0BYLQ2b", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 760.5631613505595, + "y": 468.3583537889018, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 30.123583386664713, + "height": 12.014843745913002, + "seed": 2016681117, + "groupIds": [ + "gLsxB0S7hpHJCTX-Vs7bY", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.611874996730402, + "fontFamily": 1, + "text": "secret", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "secret", + "lineHeight": 1.25, + "baseline": 9, + "index": "b0v", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 692, + "versionNonce": 1781844460, + "isDeleted": false, + "id": "TZgDiXl-Dxm_vfHo78d_O", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 816.2465724334293, + "y": 443.6974858731352, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 68.76667022705078, + "height": 11.674843976770063, + "seed": 1075871997, + "groupIds": [ + "nDtGhpyEU6AWutUYwRU6t", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.339875181416051, + "fontFamily": 1, + "text": "my-rw-password", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "my-rw-password", + "lineHeight": 1.25, + "baseline": 8, + "index": "b0w", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 2078, + "versionNonce": 398858964, + "isDeleted": false, + "id": "ULIz9z0XJ4uZmoeBz0X0J", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 804.6298664943611, + "y": 418.332609477525, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 21.024773235700497, + "height": 21.05606010063458, + "seed": 700537181, + "groupIds": [ + "1wR6eWgGRT9sfwCC6VLCm", + "ZqXqLefTE9a-uI8-aufbA", + "JKi0gRZGGGMdc1z6VKqA-", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -8.384879802332932, + 4.036005576496078 + ], + [ + -10.418526023048011, + 13.453351921653596 + ], + [ + -4.6617428751776355, + 20.930912640898267 + ], + [ + 4.661742875177646, + 21.05606010063458 + ], + [ + 10.606247212652486, + 13.422065056719518 + ], + [ + 8.541314127003327, + 4.004718711562001 + ], + [ + 0, + 0 + ] + ], + "index": "b0x", + "frameId": null + }, + { + "type": "line", + "version": 1417, + "versionNonce": 1130146924, + "isDeleted": false, + "id": "vPPUi_hnBopYQMuHHhu2L", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 804.3890997399772, + "y": 418.08040341681544, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 21.024773235700497, + "height": 21.05606010063458, + "seed": 1852319165, + "groupIds": [ + "1wR6eWgGRT9sfwCC6VLCm", + "ZqXqLefTE9a-uI8-aufbA", + "JKi0gRZGGGMdc1z6VKqA-", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -8.384879802332932, + 4.036005576496078 + ], + [ + -10.418526023048011, + 13.453351921653596 + ], + [ + -4.6617428751776355, + 20.930912640898267 + ], + [ + 4.661742875177646, + 21.05606010063458 + ], + [ + 10.606247212652486, + 13.422065056719518 + ], + [ + 8.541314127003327, + 4.004718711562001 + ], + [ + 0, + 0 + ] + ], + "index": "b0y", + "frameId": null + }, + { + "type": "rectangle", + "version": 1014, + "versionNonce": 795058260, + "isDeleted": false, + "id": "7cdqp3O7i9Oe2Tr8n7N5a", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "dotted", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 799.6166221625078, + "y": 424.21486277700797, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 9.671913041497833, + "height": 8.619434699271046, + "seed": 557213213, + "groupIds": [ + "1wR6eWgGRT9sfwCC6VLCm", + "ZqXqLefTE9a-uI8-aufbA", + "JKi0gRZGGGMdc1z6VKqA-", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b0z", + "frameId": null + }, + { + "type": "text", + "version": 919, + "versionNonce": 576995052, + "isDeleted": false, + "id": "2LbmHD87xRQM5goAMkiLX", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 802.3137957836002, + "y": 432.7721594797771, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 3.6718725876193545, + "height": 4.554023423111193, + "seed": 1964874365, + "groupIds": [ + "ZqXqLefTE9a-uI8-aufbA", + "JKi0gRZGGGMdc1z6VKqA-", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 3.6432187384889545, + "fontFamily": 1, + "text": "ns", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "ns", + "lineHeight": 1.25, + "baseline": 3, + "index": "b10", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 771, + "versionNonce": 1412724180, + "isDeleted": false, + "id": "ucVMrBYh9Viq_7WuSOW8o", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 820.6656563885158, + "y": 427.0379309173412, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 33.18778101144198, + "height": 6.754134409078216, + "seed": 1121927901, + "groupIds": [ + "JKi0gRZGGGMdc1z6VKqA-", + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 5.403307527262574, + "fontFamily": 1, + "text": "namespace-a", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "namespace-a", + "lineHeight": 1.25, + "baseline": 5, + "index": "b11", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 1110, + "versionNonce": 265650540, + "isDeleted": false, + "id": "7gOGHaziOaWEVTyYlmOtq", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 808.496835012561, + "y": 478.95265901346283, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 102.19999694824219, + "height": 64.44513875177073, + "seed": 196263667, + "groupIds": [ + "9RbhJSvGYePXc_86Xclcg" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 9.33987518141605, + "fontFamily": 2, + "text": "apiVersion: v1\nkind: Secret\nmetadata:\n name: my-rw-password\nstringData:\n password: t0ps3cr3t", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "apiVersion: v1\nkind: Secret\nmetadata:\n name: my-rw-password\nstringData:\n password: t0ps3cr3t", + "lineHeight": 1.15, + "baseline": 62, + "index": "b12", + "frameId": null, + "autoResize": true + }, + { + "id": "1YAkuXN0udd0mDd8-0X3l", + "type": "arrow", + "x": 587.435433606352, + "y": 212.09099812481958, + "width": 124.44811348963242, + "height": 125.32009325876317, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 377265053, + "version": 106, + "versionNonce": 1241459308, + "isDeleted": false, + "boundElements": [], + "updated": 1737044548301, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + -96.54346333431096, + 37.62441260495535 + ], + [ + -124.44811348963242, + 125.32009325876317 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "fWUQvR4Xdx0mfMQ7adaaJ", + "focus": 0.8190158988783627, + "gap": 12.140391573203033 + }, + "endBinding": { + "elementId": "yWwz62tI1-_nmrJMxkBgC", + "focus": -0.1513210324038822, + "gap": 16.946602734933577 + }, + "startArrowhead": null, + "endArrowhead": "triangle", + "index": "b13", + "frameId": null + }, + { + "id": "hRptkCT98fBUQz1FMt2uS", + "type": "text", + "x": 450.602328102891, + "y": 229.88752103297838, + "width": 42.79999923706055, + "height": 20, + "angle": 5.6609232054454255, + "strokeColor": "#1971c2", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 1714240829, + "version": 79, + "versionNonce": 1929016300, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "text": "needs", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "needs", + "lineHeight": 1.25, + "index": "b14", + "frameId": null, + "autoResize": true + }, + { + "id": "viZWH67LUjGqaj3n_tH0Q", + "type": "arrow", + "x": 635.3905349857018, + "y": 426.10331386300214, + "width": 138.51931375999197, + "height": 43.847840326903906, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 158091603, + "version": 57, + "versionNonce": 654840020, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 138.51931375999197, + -43.847840326903906 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "RaNUQl2ish66v6i9utUt1", + "focus": 0.3857610445013074, + "gap": 4.404230526492029 + }, + "endBinding": { + "elementId": "oi9GU0q2xYWaUaLFTOKgl", + "focus": 0.523325247627179, + "gap": 1.7665815800512519 + }, + "startArrowhead": null, + "endArrowhead": "triangle", + "index": "b15", + "frameId": null + }, + { + "id": "Osh8DpDAtYfWhIxQTpdhN", + "type": "arrow", + "x": 636.3870768113132, + "y": 432.08256481667087, + "width": 137.52277193438056, + "height": 94.67147343308801, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 1136626653, + "version": 62, + "versionNonce": 970790508, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 137.52277193438056, + 94.67147343308801 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "RaNUQl2ish66v6i9utUt1", + "focus": -0.5699782910957126, + "gap": 5.400772352103445 + }, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "index": "b16", + "frameId": null + }, + { + "type": "text", + "version": 215, + "versionNonce": 787928660, + "isDeleted": false, + "id": "IWuC3r-9N_9cmUxJHMlY2", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0.5760363230974521, + "x": 655.276719361933, + "y": 486.7548366659825, + "strokeColor": "#1971c2", + "backgroundColor": "#e9ecef", + "width": 80.76667022705078, + "height": 20, + "seed": 989311699, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "references", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "references", + "lineHeight": 1.25, + "baseline": 14, + "index": "b17", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 360, + "versionNonce": 21457132, + "isDeleted": false, + "id": "ct2jnnM3ceYjYA5cojqdC", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.9614347527829405, + "x": 655.7962512344752, + "y": 370.26238988487535, + "strokeColor": "#1971c2", + "backgroundColor": "#e9ecef", + "width": 80.76667022705078, + "height": 20, + "seed": 571981587, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "references", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "references", + "lineHeight": 1.25, + "baseline": 14, + "index": "b18", + "frameId": null, + "autoResize": true + }, + { + "id": "DYEkP80HBWsXEByJCHFDf", + "type": "ellipse", + "x": 900.4706605983483, + "y": 280.60820732372997, + "width": 28.899712942732094, + "height": 42, + "angle": 0, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 664443123, + "version": 91, + "versionNonce": 670100436, + "isDeleted": false, + "boundElements": [ + { + "type": "text", + "id": "mdvENYN1x2IYLsL1CSOQt" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b19", + "frameId": null + }, + { + "id": "mdvENYN1x2IYLsL1CSOQt", + "type": "text", + "x": 912.5362588254995, + "y": 291.75896491881247, + "width": 4.333333492279053, + "height": 20, + "angle": 0, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 167055453, + "version": 48, + "versionNonce": 1019439980, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "text": "1", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "middle", + "baseline": 14, + "containerId": "DYEkP80HBWsXEByJCHFDf", + "originalText": "1", + "lineHeight": 1.25, + "index": "b1A", + "frameId": null, + "autoResize": true + }, + { + "type": "ellipse", + "version": 136, + "versionNonce": 1946964308, + "isDeleted": false, + "id": "FD-M73iLa0obvI4Ufhrz1", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 905.9516406392113, + "y": 427.0272340264541, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "width": 28.899712942732094, + "height": 42, + "seed": 2117509203, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "zb3PkI8PEnTmliG8ko_Ep" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b1B", + "frameId": null + }, + { + "type": "text", + "version": 94, + "versionNonce": 1587224044, + "isDeleted": false, + "id": "zb3PkI8PEnTmliG8ko_Ep", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 915.8839054217672, + "y": 438.1779916215366, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "width": 8.600000381469727, + "height": 20, + "seed": 1170546163, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "1'", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "FD-M73iLa0obvI4Ufhrz1", + "originalText": "1'", + "lineHeight": 1.25, + "baseline": 14, + "index": "b1C", + "frameId": null, + "autoResize": true + }, + { + "type": "ellipse", + "version": 216, + "versionNonce": 1117718228, + "isDeleted": false, + "id": "wFQaiBWWCUoq1N9D-9Xdk", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 412.6634369615423, + "y": 284.5217529640164, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "width": 28.899712942732094, + "height": 42, + "seed": 961399795, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "22c55ojF7D4HTBt5ZOacq" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b1D", + "frameId": null + }, + { + "type": "text", + "version": 177, + "versionNonce": 2144895084, + "isDeleted": false, + "id": "22c55ojF7D4HTBt5ZOacq", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 421.19570212556795, + "y": 295.6725105590989, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "width": 11.399999618530273, + "height": 20, + "seed": 1479923091, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "2", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "wFQaiBWWCUoq1N9D-9Xdk", + "originalText": "2", + "lineHeight": 1.25, + "baseline": 14, + "index": "b1E", + "frameId": null, + "autoResize": true + }, + { + "type": "ellipse", + "version": 288, + "versionNonce": 1748962388, + "isDeleted": false, + "id": "8BPF6QKaEWQHOkLRSwGk3", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 705.6467336913092, + "y": 126.07160269179528, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "width": 28.899712942732094, + "height": 42, + "seed": 1432979613, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "Pk5TSV5zKxzb7kKAekQsy" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b1F", + "frameId": null + }, + { + "type": "text", + "version": 251, + "versionNonce": 138777324, + "isDeleted": false, + "id": "Pk5TSV5zKxzb7kKAekQsy", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 714.4289988553348, + "y": 137.22236028687777, + "strokeColor": "#e03131", + "backgroundColor": "#e9ecef", + "width": 10.899999618530273, + "height": 20, + "seed": 734555389, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "3", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "8BPF6QKaEWQHOkLRSwGk3", + "originalText": "3", + "lineHeight": 1.25, + "baseline": 14, + "index": "b1G", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 365, + "versionNonce": 918708692, + "isDeleted": false, + "id": "oqIw5jdRq7kXDexZ_elR1", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1408.54194632293, + "y": 268.04796046871735, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 508.87238614544293, + "height": 243.97521029675846, + "seed": 1589820701, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "YncFD8K1R8se9lhywuP3w", + "type": "arrow" + }, + { + "id": "4OxoKFs1_Xdbzn0FDWUjv", + "type": "arrow" + } + ], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b1H", + "frameId": null + }, + { + "id": "Dwl6y1Xw5jRc1Tu1dm9Qo", + "type": "image", + "x": 1575.594965344691, + "y": 331.44010096235445, + "width": 179.98441190983908, + "height": 114.88366717649302, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 548389757, + "version": 299, + "versionNonce": 1449940332, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "status": "saved", + "fileId": "c3a4bc93fd90fde8924b434f694db82dfa5f98a1", + "scale": [ + 1, + 1 + ], + "index": "b1I", + "frameId": null, + "crop": null + }, + { + "id": "dvxTuMmyZh2VuYCgGnakR", + "type": "text", + "x": 2182.6412144438887, + "y": 460.6988779900424, + "width": 70.93333435058594, + "height": 20, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 2021436445, + "version": 148, + "versionNonce": 1153304172, + "isDeleted": false, + "boundElements": [], + "updated": 1737044456976, + "link": null, + "locked": false, + "text": "tenant A", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "tenant A", + "lineHeight": 1.25, + "index": "b1X", + "frameId": null, + "autoResize": true + }, + { + "id": "YncFD8K1R8se9lhywuP3w", + "type": "arrow", + "x": 2170.482667876817, + "y": 402.2475607466697, + "width": 240.08323008855928, + "height": 1.4722671965227505, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 1850808285, + "version": 282, + "versionNonce": 504350956, + "isDeleted": false, + "boundElements": [], + "updated": 1737044451436, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + -240.08323008855928, + 1.4722671965227505 + ] + ], + "lastCommittedPoint": null, + "startBinding": null, + "endBinding": { + "elementId": "oqIw5jdRq7kXDexZ_elR1", + "focus": 0.12403935236777239, + "gap": 12.98510531988461 + }, + "startArrowhead": null, + "endArrowhead": "triangle", + "index": "b1Y", + "frameId": null + }, + { + "id": "4OxoKFs1_Xdbzn0FDWUjv", + "type": "arrow", + "x": 1126.5772342692233, + "y": 11.389001324940125, + "width": 279.7331204728548, + "height": 400.7895823551674, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 1979849949, + "version": 319, + "versionNonce": 1343426260, + "isDeleted": false, + "boundElements": [], + "updated": 1737044579417, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 60.077003673679656, + 384.9981342659643 + ], + [ + 279.7331204728548, + 400.7895823551674 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "5cjKeARw4GNFEEJZ-9VN4", + "focus": -1.1565162221439549, + "gap": 10.916770203977023 + }, + "endBinding": { + "elementId": "oqIw5jdRq7kXDexZ_elR1", + "focus": -0.289388765386766, + "gap": 2.231591580851841 + }, + "startArrowhead": null, + "endArrowhead": "triangle", + "index": "b1Z", + "frameId": null + }, + { + "id": "OaNLM_wXNAsD2JNP3d5Id", + "type": "text", + "x": 1220.4384787009835, + "y": 372.8300033237059, + "width": 167.60000610351562, + "height": 20, + "angle": 0.027372567054417374, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 690672541, + "version": 479, + "versionNonce": 2074024940, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "text": "Metrics for tenant A", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "Metrics for tenant A", + "lineHeight": 1.25, + "index": "b1a", + "frameId": null, + "autoResize": true + }, + { + "id": "PLgEMMqivsgP0dLEpzz8O", + "type": "line", + "x": 627.1612461803114, + "y": 256.4215539372852, + "width": 299.1550371768452, + "height": 0, + "angle": 0, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 2 + }, + "seed": 1946693885, + "version": 48, + "versionNonce": 1717692628, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 299.1550371768452, + 0 + ] + ], + "lastCommittedPoint": null, + "startBinding": null, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": null, + "index": "b1b", + "frameId": null + }, + { + "id": "aENYzNbXqw33nLwOf4wyP", + "type": "rectangle", + "x": 299.68383861985967, + "y": 419.2751836430234, + "width": 157.54318525881195, + "height": 95.58800004467241, + "angle": 0, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": { + "type": 3 + }, + "seed": 32737789, + "version": 78, + "versionNonce": 397196908, + "isDeleted": false, + "boundElements": [], + "updated": 1737044362106, + "link": null, + "locked": false, + "index": "b1c", + "frameId": null + }, + { + "type": "line", + "version": 1620, + "versionNonce": 1282638700, + "isDeleted": false, + "id": "ZZ-5a8rbArSN1llGHNXh4", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1794.3367828327116, + "y": 107.57017147010117, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 49.26942071813747, + "height": 43.87300421060919, + "seed": 1776200940, + "groupIds": [ + "_m7UQDmuNP__-mwBLliPJ", + "dHILZpKzvGpxdJTDbqx_K" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1737044448319, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.518175120431392, + -29.072472669680792 + ], + [ + 23.649321944705985, + -43.87300421060919 + ], + [ + 41.780468768980576, + -32.244015142736835 + ], + [ + 49.26942071813747, + -3.0188078795298896 + ], + [ + 0, + 0 + ] + ], + "index": "b1e", + "frameId": null, + "roundness": { + "type": 2 + } + }, + { + "type": "ellipse", + "version": 1260, + "versionNonce": 814233068, + "isDeleted": false, + "id": "3OiGqb0D4v69xNMrBOHVk", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1806.7906536690273, + "y": 40.879123684244746, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 25.225943407686408, + "height": 22.072700481725683, + "seed": 433205100, + "groupIds": [ + "_m7UQDmuNP__-mwBLliPJ", + "dHILZpKzvGpxdJTDbqx_K" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1737044448319, + "link": null, + "locked": false, + "index": "b1f", + "frameId": null, + "roundness": null + }, + { + "type": "line", + "version": 1709, + "versionNonce": 960269036, + "isDeleted": false, + "id": "GYXw6b97EZEfZ3ABbxyoz", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2189.993032832712, + "y": 443.5384342753816, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 66.25527828948614, + "height": 58.9984225712568, + "seed": 215534036, + "groupIds": [ + "WKqo2SzDOPQ75oD8FzEY_", + "NCeRImamlfN_QQU7b58_c" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1737044454793, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 7.420591168422442, + -39.09534025806174 + ], + [ + 31.802533578953348, + -58.9984225712568 + ], + [ + 56.184475989484255, + -43.36028646803206 + ], + [ + 66.25527828948614, + -4.059555668514616 + ], + [ + 0, + 0 + ] + ], + "index": "b1g", + "frameId": null, + "roundness": { + "type": 2 + } + }, + { + "type": "ellipse", + "version": 1349, + "versionNonce": 1859578324, + "isDeleted": false, + "id": "pxP5BVaGklSqiGtNtMiLK", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2206.7404324573563, + "y": 353.85534350498995, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 33.922702484216934, + "height": 29.68236467368992, + "seed": 1816686420, + "groupIds": [ + "WKqo2SzDOPQ75oD8FzEY_", + "NCeRImamlfN_QQU7b58_c" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1737044454793, + "link": null, + "locked": false, + "index": "b1h", + "frameId": null, + "roundness": null + } + ], + "appState": { + "gridSize": 20, + "gridStep": 5, + "gridModeEnabled": false, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "0644a8702c31871ef24e831da4f3ed873aa2d825": { + "mimeType": "image/png", + "id": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "dataURL": "", + "created": 1726574587466, + "lastRetrieved": 1737043646801 + }, + "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78": { + "mimeType": "image/png", + "id": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "dataURL": "", + "created": 1726574587467, + "lastRetrieved": 1737043646801 + }, + "1493a0629beacbb7e4599ef52f315c4c7b9f2307": { + "mimeType": "image/png", + "id": "1493a0629beacbb7e4599ef52f315c4c7b9f2307", + "dataURL": "", + "created": 1726574587474, + "lastRetrieved": 1737043646801 + }, + "c3a4bc93fd90fde8924b434f694db82dfa5f98a1": { + "mimeType": "image/png", + "id": "c3a4bc93fd90fde8924b434f694db82dfa5f98a1", + "dataURL": "", + "created": 1686150806191, + "lastRetrieved": 1686150806191 + } + } +} \ No newline at end of file diff --git a/docs/images/source/send-metrics-grafana-cloud.excalidraw b/docs/images/source/send-metrics-grafana-cloud.excalidraw new file mode 100644 index 0000000..000aa97 --- /dev/null +++ b/docs/images/source/send-metrics-grafana-cloud.excalidraw @@ -0,0 +1,2850 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "rectangle", + "version": 755, + "versionNonce": 518776300, + "isDeleted": false, + "id": "E0_cqgLTCZYdblXeBNRce", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 246.98137933537976, + "y": -118.75289187826343, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 953.4529315439731, + "height": 402.5133833897971, + "seed": 1175240413, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "a0", + "frameId": null + }, + { + "type": "text", + "version": 254, + "versionNonce": 1018260716, + "isDeleted": false, + "id": "q_2M1iXS5kZHhdte9Lik-", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 298.18949003310775, + "y": -162.9568837253754, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 121.13990294933319, + "height": 25, + "seed": 1258711965, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044650686, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "K8s Cluster", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "K8s Cluster", + "lineHeight": 1.25, + "baseline": 18, + "index": "a2", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 243, + "versionNonce": 2139604052, + "isDeleted": false, + "id": "Pf29U31FXugJR3y16teDk", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 411.6691523387309, + "y": -87.01816200445023, + "strokeColor": "#000000", + "backgroundColor": "#a5d8ff", + "width": 229.49400491361894, + "height": 134.93853507796484, + "seed": 1104211965, + "groupIds": [ + "2xLoSSjPhMgeqTcoAJ45m" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "a3", + "frameId": null + }, + { + "type": "rectangle", + "version": 366, + "versionNonce": 1896110828, + "isDeleted": false, + "id": "W7YGziUK06CJHuE-QzOK2", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 415.11648717648916, + "y": -83.5708271666922, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 222.5993352381026, + "height": 44.32287648546298, + "seed": 428599389, + "groupIds": [ + "2xLoSSjPhMgeqTcoAJ45m" + ], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "a4", + "frameId": null + }, + { + "type": "text", + "version": 315, + "versionNonce": 1412700628, + "isDeleted": false, + "id": "TBkpSzAa8wHxnqT9E5CH2", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 433.3381141760684, + "y": -72.24386984262946, + "strokeColor": "#1e1e1e", + "backgroundColor": "#1971c2", + "width": 187.93333435058594, + "height": 21.507525856732585, + "seed": 1141429437, + "groupIds": [ + "2xLoSSjPhMgeqTcoAJ45m" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 17.20602068538607, + "fontFamily": 1, + "text": "Namespace A (tenant)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Namespace A (tenant)", + "lineHeight": 1.25, + "baseline": 15, + "index": "a5", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1649, + "versionNonce": 1276255596, + "isDeleted": false, + "id": "evHzKLVy0EOUfnX7mLgwn", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 465.62278026704223, + "y": -31.678821222182478, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 593366589, + "groupIds": [ + "d3WFboVCh_A2k7YVjEh9d", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ], + "index": "a6", + "frameId": null + }, + { + "type": "line", + "version": 988, + "versionNonce": 1270464340, + "isDeleted": false, + "id": "q-_In0nWigaA4NFversey", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 464.9287401175453, + "y": -32.40583659657216, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 60.606526831303974, + "height": 60.696715115279126, + "seed": 5979805, + "groupIds": [ + "d3WFboVCh_A2k7YVjEh9d", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -24.17046010534145, + 11.634288632794958 + ], + [ + -30.032698563726516, + 38.78096210931654 + ], + [ + -13.438054312298041, + 60.33596197937851 + ], + [ + 13.438054312298071, + 60.696715115279126 + ], + [ + 30.573828267577458, + 38.69077382534138 + ], + [ + 24.621401525217237, + 11.544100348819807 + ], + [ + 0, + 0 + ] + ], + "index": "a7", + "frameId": null + }, + { + "type": "line", + "version": 685, + "versionNonce": 893620204, + "isDeleted": false, + "id": "WF4Fm2n3yzQKdphnhCcfQ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 452.9320826343916, + "y": -9.685450544942455, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 11.606398134643495, + "height": 20.43580530455634, + "seed": 991126269, + "groupIds": [ + "IEPHLG7g-hTwmKjiyyTby", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.10680734479733643, + 14.205376858045264 + ], + [ + 11.321578548517282, + 20.43580530455634 + ], + [ + 11.606398134643495, + 3.8806668609697628 + ], + [ + 1.3528930340995515, + 1.032470999707551 + ] + ], + "index": "a8", + "frameId": null + }, + { + "type": "line", + "version": 903, + "versionNonce": 1584313556, + "isDeleted": false, + "id": "7fHaJR-Tl4E0RzwR8kL4r", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 478.3522306961569, + "y": -9.756655441473981, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 12.667312471753975, + "height": 20.509145060610187, + "seed": 1208482653, + "groupIds": [ + "IEPHLG7g-hTwmKjiyyTby", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -12.667312471753975, + 3.6435687166298947 + ], + [ + -11.887638165338293, + 20.509145060610187 + ], + [ + -0.5696391722524426, + 14.347786651108377 + ], + [ + 0, + 0 + ] + ], + "index": "a9", + "frameId": null + }, + { + "type": "line", + "version": 618, + "versionNonce": 211989100, + "isDeleted": false, + "id": "FnQi7TijNDGbvWKKbar2z", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 453.8577462893019, + "y": -11.32316316516824, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 24.458881958589213, + "height": 7.5121165840790765, + "seed": 260736957, + "groupIds": [ + "IEPHLG7g-hTwmKjiyyTby", + "_-fMpR10t-Cc1HTfdpw7u" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.962422617301264, + 3.4890399300462054 + ], + [ + 24.458881958589213, + 0.28481958612622416 + ], + [ + 11.855615272503934, + -4.023076654032871 + ], + [ + 0, + 0 + ] + ], + "index": "aA", + "frameId": null + }, + { + "type": "image", + "version": 1117, + "versionNonce": 166263380, + "isDeleted": false, + "id": "51P_ipCxt3SGjchRBgDYJ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1301.6185391102308, + "y": -228.15461704591326, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 109.62509643488907, + "height": 109.62509643488907, + "seed": 1478862653, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "scale": [ + 1, + 1 + ], + "index": "aB", + "frameId": null, + "crop": null + }, + { + "type": "rectangle", + "version": 1179, + "versionNonce": 553223404, + "isDeleted": false, + "id": "qCLTpBrh4o5Fh5Rs7SoIk", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1299.4710109345306, + "y": -124.2426560040185, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 543.8990028571536, + "height": 266.96980059334186, + "seed": 1254602653, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "aC", + "frameId": null + }, + { + "type": "rectangle", + "version": 896, + "versionNonce": 687771604, + "isDeleted": false, + "id": "I7fko55wggHsMjE_Zjhfo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1369.882421859406, + "y": -49.45720402190682, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 404.2320808000759, + "height": 122.7462711105253, + "seed": 286283133, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "nUi8CkYcll8R9p3URXDIG", + "type": "arrow" + }, + { + "id": "e-8ZsFtNd0YWIHs4OXK9I", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "aD", + "frameId": null + }, + { + "type": "image", + "version": 1084, + "versionNonce": 1473382252, + "isDeleted": false, + "id": "2Fc9c2IqFS8qhO9nP84CT", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1491.233833409322, + "y": -23.679340890517096, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 1724030429, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "index": "aE", + "frameId": null, + "crop": null + }, + { + "type": "text", + "version": 863, + "versionNonce": 1271231828, + "isDeleted": false, + "id": "1rN_kHhApPEKx2EMm4MbX", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1379.567168014221, + "y": -84.1793408905171, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 88.66666412353516, + "height": 25, + "seed": 1896909469, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "tenant A", + "lineHeight": 1.25, + "baseline": 18, + "index": "aF", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 1358, + "versionNonce": 1433811436, + "isDeleted": false, + "id": "GEPrKk2_r33AcFZt-4owF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 985.0696693817434, + "y": -51.79276395198133, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 91.52638459323295, + "seed": 2134847229, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + }, + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "aG", + "frameId": null + }, + { + "type": "rectangle", + "version": 1492, + "versionNonce": 261335764, + "isDeleted": false, + "id": "KVhxDF9yv_MWK13J3dHL3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 987.4079346815705, + "y": -49.45449865215426, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 752502621, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "aH", + "frameId": null + }, + { + "type": "text", + "version": 1499, + "versionNonce": 1255983596, + "isDeleted": false, + "id": "R1JboiRKt3oNY9o9xC077", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1011.8681011442326, + "y": -41.77162695272227, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 1009996733, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044662200, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aI", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 795, + "versionNonce": 383798100, + "isDeleted": false, + "id": "5cjKeARw4GNFEEJZ-9VN4", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1012.4045360256955, + "y": -5.529571655364862, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 106.99192810058594, + "height": 40, + "seed": 694932509, + "groupIds": [ + "yAcWBuG02Vk1oi-K2pIB0" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044681933, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Observability \nOperator*", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Observability \nOperator*", + "lineHeight": 1.25, + "baseline": 34, + "index": "aJ", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 1102, + "versionNonce": 1611247340, + "isDeleted": false, + "id": "maCvw5JncD4BKuAguGF0c", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 717.0696693817434, + "y": -94.19780849054655, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 155.66166138849098, + "height": 141.52638459323293, + "seed": 1886734461, + "groupIds": [ + "BSJ9NpvgfUQydZ2CrVzeX", + "RUi1FtLYA-77GfM1YBh0Y" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "jqn7L1CPT_HyliDPI9P-j", + "type": "arrow" + }, + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "aK", + "frameId": null + }, + { + "type": "rectangle", + "version": 1200, + "versionNonce": 1105069524, + "isDeleted": false, + "id": "dvmQ_NYaRcNJZOZm3v1OF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 719.4079346815705, + "y": -91.85954319071948, + "strokeColor": "#1971c2", + "backgroundColor": "#1971c2", + "width": 150.98513078883684, + "height": 38.60754094432087, + "seed": 319893725, + "groupIds": [ + "BSJ9NpvgfUQydZ2CrVzeX", + "RUi1FtLYA-77GfM1YBh0Y" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "nUi8CkYcll8R9p3URXDIG", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "aL", + "frameId": null + }, + { + "type": "text", + "version": 1113, + "versionNonce": 1034647892, + "isDeleted": false, + "id": "l7HSSaoTOrfjhrJnj8QIl", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 746.6337261442326, + "y": -86.1766714912875, + "strokeColor": "#ffffff", + "backgroundColor": "#1971c2", + "width": 103.34941101074219, + "height": 29.176336945926444, + "seed": 1590758717, + "groupIds": [ + "RUi1FtLYA-77GfM1YBh0Y" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044655913, + "link": null, + "locked": false, + "fontSize": 11.670534778370577, + "fontFamily": 1, + "text": "platform-services \nNamespace", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "platform-services \nNamespace", + "lineHeight": 1.25, + "baseline": 25, + "index": "aM", + "frameId": null, + "autoResize": true + }, + { + "type": "image", + "version": 654, + "versionNonce": 1162226516, + "isDeleted": false, + "id": "YwqW6r2OmqYpQ8qE_dmeC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 756.686885052514, + "y": -42.179340890517096, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 78.42723004694828, + "height": 64.99999999999999, + "seed": 1930845597, + "groupIds": [ + "6-seN9eTnZZzWPC82Jr-R", + "pWqSYLnDuN9C5Ko6KXKh1", + "OQDO0bZJrfwwUc2POj11y" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "status": "saved", + "fileId": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "scale": [ + 1, + 1 + ], + "index": "aN", + "frameId": null, + "crop": null + }, + { + "type": "arrow", + "version": 446, + "versionNonce": 973333484, + "isDeleted": false, + "id": "jqn7L1CPT_HyliDPI9P-j", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 715.9005000759886, + "y": -10.78325670726747, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 220, + "height": 17.1039158167504, + "seed": 897006461, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": { + "elementId": "maCvw5JncD4BKuAguGF0c", + "focus": -0.08474347907800747, + "gap": 1.1691693057548491 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -220, + 17.1039158167504 + ] + ], + "index": "aO", + "frameId": null + }, + { + "type": "text", + "version": 179, + "versionNonce": 1133355220, + "isDeleted": false, + "id": "r9D6VpvKhMXXNF5ZT-yj8", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 509.567168014221, + "y": -30.679340890517096, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 122.66666412353516, + "height": 20, + "seed": 1767687133, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "scrapes metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "scrapes metrics", + "lineHeight": 1.25, + "baseline": 14, + "index": "aP", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 161, + "versionNonce": 1043891820, + "isDeleted": false, + "id": "TErMRO_X_1wfaHA1LWMjA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 751.350500838928, + "y": 20.320659109482904, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 103.0999984741211, + "height": 20, + "seed": 1861251901, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "prometheus-A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "prometheus-A", + "lineHeight": 1.25, + "baseline": 14, + "index": "aQ", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 411, + "versionNonce": 1145726548, + "isDeleted": false, + "id": "QwDloiGA_JLdFTmuU7unS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 975.4699826731477, + "y": -3.436100817368242, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 98.5694825971591, + "height": 4.3436520113717165, + "seed": 1580191005, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": { + "elementId": "KdeBQ3YHl9MqzPsBoCQuZ", + "focus": -1.9242239307487354, + "gap": 13.795993107279287 + }, + "endBinding": { + "elementId": "maCvw5JncD4BKuAguGF0c", + "focus": 0.16229811829395357, + "gap": 4.169169305754167 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -98.5694825971591, + -4.3436520113717165 + ] + ], + "index": "aR", + "frameId": null + }, + { + "type": "text", + "version": 196, + "versionNonce": 1219811564, + "isDeleted": false, + "id": "KdeBQ3YHl9MqzPsBoCQuZ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 902.8785583602222, + "y": -37.23209392464753, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 66.53333282470703, + "height": 20, + "seed": 1609230813, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "QwDloiGA_JLdFTmuU7unS", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Manages", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Manages", + "lineHeight": 1.25, + "baseline": 14, + "index": "aS", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 290, + "versionNonce": 1817344980, + "isDeleted": false, + "id": "nUi8CkYcll8R9p3URXDIG", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 878.361346023834, + "y": -73.26531266693223, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 670.1137683788025, + "height": 158.0800364665438, + "seed": 854269981, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": { + "elementId": "dvmQ_NYaRcNJZOZm3v1OF", + "focus": 0.7008811292253843, + "gap": 7.968280553426666 + }, + "endBinding": { + "elementId": "I7fko55wggHsMjE_Zjhfo", + "focus": 0.3701488995551903, + "gap": 2.647788803047206 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 293.75502839452247, + -136.91971662456558 + ], + [ + 670.1137683788025, + 21.160319841978207 + ] + ], + "index": "aT", + "frameId": null + }, + { + "type": "text", + "version": 633, + "versionNonce": 352508780, + "isDeleted": false, + "id": "-OhjoaqmN_5bGegwBeSwA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.256875589926665, + "x": 1060.034724783709, + "y": -172.9843439377319, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 209.5, + "height": 25, + "seed": 606803069, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Metrics for tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Metrics for tenant A", + "lineHeight": 1.25, + "baseline": 18, + "index": "aU", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 866, + "versionNonce": 33896020, + "isDeleted": false, + "id": "NhFzqMhqC7Wd9-fMvGCJE", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1983.0958039087182, + "y": 46.79251030116768, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 74.76667022705078, + "height": 20, + "seed": 1307500381, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044756645, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Tenant A", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Tenant A", + "lineHeight": 1.25, + "baseline": 14, + "index": "aj", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 1888, + "versionNonce": 2103104084, + "isDeleted": false, + "id": "e-8ZsFtNd0YWIHs4OXK9I", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1966.3725730818032, + "y": 1.0320115136515202, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 188.21715310427362, + "height": 11.039260063766374, + "seed": 1907417213, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044743225, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "I7fko55wggHsMjE_Zjhfo", + "focus": 0.16611802797687503, + "gap": 4.040917318047832 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -188.21715310427362, + 11.039260063766374 + ] + ], + "index": "ak", + "frameId": null + }, + { + "type": "text", + "version": 457, + "versionNonce": 1904627028, + "isDeleted": false, + "id": "YWCn7LyM3XG-reKjncQ4J", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1484.510380338059, + "y": 158.60257174082597, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 354.8333435058594, + "height": 40, + "seed": 421120221, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "** It is also valid for other remote writes, \nsuch as Victoria Metrics", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "** It is also valid for other remote writes, \nsuch as Victoria Metrics", + "lineHeight": 1.25, + "baseline": 34, + "index": "al", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 621, + "versionNonce": 181330412, + "isDeleted": false, + "id": "8L8CgMD_SJbwW4m3ZHH1h", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1388.406549673086, + "y": -222.18259137152484, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 41.266666412353516, + "height": 49.957396413008695, + "seed": 1208395069, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 39.96591713040696, + "fontFamily": 1, + "text": "**", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "**", + "lineHeight": 1.25, + "baseline": 35, + "index": "am", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 768, + "versionNonce": 1199344492, + "isDeleted": false, + "id": "0CN7aawE0LBj0L0-wH0AS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 590.5544491096351, + "y": 172.18629390870842, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 310.9500714804964, + "height": 94.57612411084251, + "seed": 1308409811, + "groupIds": [ + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + } + ], + "updated": 1737044725728, + "link": null, + "locked": false, + "index": "an", + "frameId": null + }, + { + "type": "text", + "version": 642, + "versionNonce": 1602501612, + "isDeleted": false, + "id": "fWUQvR4Xdx0mfMQ7adaaJ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 598.7512785795676, + "y": 188.25805356153091, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 245.23074531555176, + "height": 64.44513875177073, + "seed": 181975091, + "groupIds": [ + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044735311, + "link": null, + "locked": false, + "fontSize": 9.33987518141605, + "fontFamily": 2, + "text": "apiVersion: v1\nkind: Namespace\nmetadata:\n// Ommitted for brevity\n annotations:\n grafanacloud.adevinta.com/stack-name: stack1", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "apiVersion: v1\nkind: Namespace\nmetadata:\n// Ommitted for brevity\n annotations:\n grafanacloud.adevinta.com/stack-name: stack1", + "lineHeight": 1.15, + "baseline": 62, + "index": "ao", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1625, + "versionNonce": 879555564, + "isDeleted": false, + "id": "zUhv0FUCvdgCgSAyodVPH", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 593.760334927418, + "y": 113.0465891846303, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 67.37474875735316, + "height": 67.47500880014684, + "seed": 220998355, + "groupIds": [ + "21zmTFQ4XmHnDWJaDcdr-", + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz", + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044719688, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -26.869691468706304, + 12.933545520384756 + ], + [ + -33.38659425029553, + 43.111818401282534 + ], + [ + -14.93874637625835, + 67.07396862897212 + ], + [ + 14.938746376258385, + 67.47500880014684 + ], + [ + 33.98815450705763, + 43.011558358488855 + ], + [ + 27.37099168267472, + 12.83328547759108 + ], + [ + 0, + 0 + ] + ], + "index": "ap", + "frameId": null + }, + { + "type": "line", + "version": 964, + "versionNonce": 1274908884, + "isDeleted": false, + "id": "Q1fmEtTedxlYJAAmyhdyr", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 592.9887879811621, + "y": 112.23838451174788, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 67.37474875735316, + "height": 67.47500880014684, + "seed": 581261427, + "groupIds": [ + "21zmTFQ4XmHnDWJaDcdr-", + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz", + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044719688, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -26.869691468706304, + 12.933545520384756 + ], + [ + -33.38659425029553, + 43.111818401282534 + ], + [ + -14.93874637625835, + 67.07396862897212 + ], + [ + 14.938746376258385, + 67.47500880014684 + ], + [ + 33.98815450705763, + 43.011558358488855 + ], + [ + 27.37099168267472, + 12.83328547759108 + ], + [ + 0, + 0 + ] + ], + "index": "aq", + "frameId": null + }, + { + "type": "rectangle", + "version": 530, + "versionNonce": 687838828, + "isDeleted": false, + "id": "KH4s95NUqWRw2hUYctDtG", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "dotted", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 577.6951877065052, + "y": 131.89651130920402, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 30.994042307547033, + "height": 27.621332262823326, + "seed": 937939475, + "groupIds": [ + "21zmTFQ4XmHnDWJaDcdr-", + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz", + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044719688, + "link": null, + "locked": false, + "index": "ar", + "frameId": null + }, + { + "type": "text", + "version": 467, + "versionNonce": 423384660, + "isDeleted": false, + "id": "CzfYIVdX3lo7Vn9MaQchT", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 586.3383912758778, + "y": 159.31871981189613, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 11.766666412353516, + "height": 14.593554970962579, + "seed": 501880755, + "groupIds": [ + "GNsaZUttB4FcuDkwwe2Ds", + "3q-izW3IfC3KM17_DJnbz", + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044719688, + "link": null, + "locked": false, + "fontSize": 11.674843976770063, + "fontFamily": 1, + "text": "ns", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "ns", + "lineHeight": 1.25, + "baseline": 10, + "index": "as", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 288, + "versionNonce": 1746528492, + "isDeleted": false, + "id": "nYzmlP6fDWXWoYoi0xwJC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 636.9152942620094, + "y": 142.51535301118827, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 57.36666488647461, + "height": 11.674843976770063, + "seed": 1401292979, + "groupIds": [ + "3q-izW3IfC3KM17_DJnbz", + "vtWcKOuL00F1N4wAFZa-b" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044719688, + "link": null, + "locked": false, + "fontSize": 9.339875181416051, + "fontFamily": 1, + "text": "namespace-a", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "namespace-a", + "lineHeight": 1.25, + "baseline": 8, + "index": "at", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 705, + "versionNonce": 1966095852, + "isDeleted": false, + "id": "_RKMDF_6DBzrjhTiwqqgg", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 996.5010487494209, + "y": 51.35153769937449, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 133.13419093431673, + "height": 119.53456918749555, + "seed": 1568120189, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044725728, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Nl0-u8ODET-dMN_wpF6ze", + "focus": 1.5858518489542759, + "gap": 13.004679288878606 + }, + "endBinding": { + "elementId": "0CN7aawE0LBj0L0-wH0AS", + "focus": 0.30373891432289296, + "gap": 1.3001870218383829 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -133.13419093431673, + 119.53456918749555 + ] + ], + "index": "au", + "frameId": null + }, + { + "type": "text", + "version": 437, + "versionNonce": 2110224596, + "isDeleted": false, + "id": "Nl0-u8ODET-dMN_wpF6ze", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 5.625296701997378, + "x": 856.1734049807727, + "y": 108.06439485700457, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "width": 183.23333740234375, + "height": 20, + "seed": 1431953427, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "_RKMDF_6DBzrjhTiwqqgg", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Reads NS configuration", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Reads NS configuration", + "lineHeight": 1.25, + "baseline": 14, + "index": "av", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 208, + "versionNonce": 1059597932, + "isDeleted": false, + "id": "Hbs8m9uSj0pJIolvMgyGQ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 623.7561960591695, + "y": 258.90264885644615, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 241.43700685618546, + "height": 1.0589342405974094, + "seed": 1883590067, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 241.43700685618546, + -1.0589342405974094 + ] + ], + "index": "aw", + "frameId": null + }, + { + "type": "rectangle", + "version": 727, + "versionNonce": 1233731156, + "isDeleted": false, + "id": "7FyPlkttmsqS1wAmduadV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -102.79613340713274, + "y": 28.104441153417156, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 283.1102277304964, + "height": 123.09322279542529, + "seed": 1943883379, + "groupIds": [ + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "1A6hmLOM_Al8UdArgX_ED", + "type": "arrow" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "ax", + "frameId": null + }, + { + "type": "text", + "version": 609, + "versionNonce": 1567588588, + "isDeleted": false, + "id": "yT6DJj4so-1nw75Dr3EpQ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -94.59930393720029, + "y": 44.17620080623965, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 146.46665954589844, + "height": 85.92685166902764, + "seed": 1253115923, + "groupIds": [ + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 9.33987518141605, + "fontFamily": 2, + "text": "apiVersion: v1\nkind: Pod\nmetadata:\n// Ommitted for brevity\n annotations:\n prometheus.io/path: /_/metrics\n prometheus.io/port: \"9090\"\n prometheus.io/scrape: \"true\"", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "apiVersion: v1\nkind: Pod\nmetadata:\n// Ommitted for brevity\n annotations:\n prometheus.io/path: /_/metrics\n prometheus.io/port: \"9090\"\n prometheus.io/scrape: \"true\"", + "lineHeight": 1.15, + "baseline": 83, + "index": "ay", + "frameId": null, + "autoResize": true + }, + { + "type": "line", + "version": 1632, + "versionNonce": 334910420, + "isDeleted": false, + "id": "bw8Qym1pTZ5O4eJ5kmsvt", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -95.20792960091606, + "y": -21.82996607638256, + "strokeColor": "#aaa", + "backgroundColor": "transparent", + "width": 59.77373985289752, + "height": 59.86268887053576, + "seed": 420087997, + "groupIds": [ + "_eg-bi5Gn3W_-bYAFxzfL", + "9HfuV83kMnAX63cyRDiKA", + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -23.8383367270484, + 11.474423275333004 + ], + [ + -29.620022873534033, + 38.24807758444336 + ], + [ + -13.253403628097802, + 59.5068927999828 + ], + [ + 13.25340362809783, + 59.86268887053576 + ], + [ + 30.153716979363487, + 38.15912856680512 + ], + [ + 24.283081815239616, + 11.385474257694767 + ], + [ + 0, + 0 + ] + ], + "index": "az", + "frameId": null + }, + { + "type": "line", + "version": 971, + "versionNonce": 203981676, + "isDeleted": false, + "id": "sJoQARdvsCcP82cYkxAIS", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -95.89243302839282, + "y": -22.546991620162316, + "strokeColor": "#fff", + "backgroundColor": "#228be6", + "width": 59.77373985289752, + "height": 59.86268887053576, + "seed": 61073693, + "groupIds": [ + "_eg-bi5Gn3W_-bYAFxzfL", + "9HfuV83kMnAX63cyRDiKA", + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -23.8383367270484, + 11.474423275333004 + ], + [ + -29.620022873534033, + 38.24807758444336 + ], + [ + -13.253403628097802, + 59.5068927999828 + ], + [ + 13.25340362809783, + 59.86268887053576 + ], + [ + 30.153716979363487, + 38.15912856680512 + ], + [ + 24.283081815239616, + 11.385474257694767 + ], + [ + 0, + 0 + ] + ], + "index": "b00", + "frameId": null + }, + { + "type": "line", + "version": 689, + "versionNonce": 186956116, + "isDeleted": false, + "id": "E_Ynrilrtjji0rdpYxSP9", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -107.72424588739912, + "y": -0.13880365407590034, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 11.446916017154132, + "height": 20.154999367627223, + "seed": 1424145789, + "groupIds": [ + "sXrV9Am2N5QYk6FR8rdm4", + "9HfuV83kMnAX63cyRDiKA", + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.10533971794927477, + 14.010182487253072 + ], + [ + 11.16601010262275, + 20.154999367627223 + ], + [ + 11.446916017154132, + 3.8273430854901918 + ], + [ + 1.3343030940241043, + 1.0182839401762889 + ] + ], + "index": "b01", + "frameId": null + }, + { + "type": "line", + "version": 886, + "versionNonce": 832386540, + "isDeleted": false, + "id": "bqak-gPodd-CKPzT6pHL6", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -82.6533930154728, + "y": -0.20903013270867632, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 12.493252458263255, + "height": 20.227331370935048, + "seed": 1409811933, + "groupIds": [ + "sXrV9Am2N5QYk6FR8rdm4", + "9HfuV83kMnAX63cyRDiKA", + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -12.493252458263255, + 3.5935028781669116 + ], + [ + -11.72429156249373, + 20.227331370935048 + ], + [ + -0.5618118290627808, + 14.15063544451877 + ], + [ + 0, + 0 + ] + ], + "index": "b02", + "frameId": null + }, + { + "type": "line", + "version": 622, + "versionNonce": 711618260, + "isDeleted": false, + "id": "PNL_l1FxQZftpyeWKiqDj", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -106.8113016651721, + "y": -1.7540126626315669, + "strokeColor": "#228be6", + "backgroundColor": "#fff", + "width": 24.12279541038311, + "height": 7.40889349576541, + "seed": 1680163389, + "groupIds": [ + "sXrV9Am2N5QYk6FR8rdm4", + "9HfuV83kMnAX63cyRDiKA", + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.798048410318366, + 3.4410974530095264 + ], + [ + 24.12279541038311, + 0.2809059145313932 + ], + [ + 11.692708692369097, + -3.967796042755884 + ], + [ + 0, + 0 + ] + ], + "index": "b03", + "frameId": null + }, + { + "type": "text", + "version": 414, + "versionNonce": 1524078700, + "isDeleted": false, + "id": "AVsXWyToFRoD7PXeOKzvg", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -103.74534284300893, + "y": 19.576778625539873, + "strokeColor": "#fff", + "backgroundColor": "transparent", + "width": 16.733333587646484, + "height": 12.947155639939574, + "seed": 169603741, + "groupIds": [ + "9HfuV83kMnAX63cyRDiKA", + "e8kr6fNA1ym26CazRLzCR" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 10.357724511951659, + "fontFamily": 1, + "text": "pod", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "pod", + "lineHeight": 1.25, + "baseline": 9, + "index": "b04", + "frameId": null, + "autoResize": true + }, + { + "type": "arrow", + "version": 217, + "versionNonce": 308114516, + "isDeleted": false, + "id": "1A6hmLOM_Al8UdArgX_ED", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 443.2604419935551, + "y": 6.20375239799705, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 262.7646950222262, + "height": 73.32968233178394, + "seed": 2018502675, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "7FyPlkttmsqS1wAmduadV", + "focus": 0.21571170056017522, + "gap": 1 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + -139.5300899924224, + -42.77564802687402 + ], + [ + -262.7646950222262, + 30.55403430490992 + ] + ], + "index": "b05", + "frameId": null + }, + { + "type": "text", + "version": 194, + "versionNonce": 1671539436, + "isDeleted": false, + "id": "pUd5NA4yFpiLwMjv_EWv1", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 6.041766551995441, + "x": 191.69889288312925, + "y": -52.867380591495476, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 135.6666717529297, + "height": 20, + "seed": 107251325, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Pod configuration", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Pod configuration", + "lineHeight": 1.25, + "baseline": 14, + "index": "b06", + "frameId": null, + "autoResize": true + }, + { + "type": "rectangle", + "version": 162, + "versionNonce": 1624492500, + "isDeleted": false, + "id": "1H8BL1pqaR7gzrWpXuVHe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -81.25048024073374, + "y": 96.84738750256338, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 142.5854934229135, + "height": 44.81258364720157, + "seed": 1922583827, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "b07", + "frameId": null + }, + { + "type": "ellipse", + "version": 430, + "versionNonce": 512060780, + "isDeleted": false, + "id": "4oSwvWNFM2IQtkmYz4iAH", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 708.2111163320881, + "y": 120.7357892517, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 36, + "height": 42, + "seed": 1108832733, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "Q7ugeadK4bBZ5ZLBTtwIi" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "b08", + "frameId": null + }, + { + "type": "text", + "version": 561, + "versionNonce": 149122900, + "isDeleted": false, + "id": "Q7ugeadK4bBZ5ZLBTtwIi", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 723.8165275245907, + "y": 131.8865468467825, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 4.333333492279053, + "height": 20, + "seed": 1781674397, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "1", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "4oSwvWNFM2IQtkmYz4iAH", + "originalText": "1", + "lineHeight": 1.25, + "baseline": 14, + "index": "b09", + "frameId": null, + "autoResize": true + }, + { + "type": "text", + "version": 290, + "versionNonce": 2093801452, + "isDeleted": false, + "id": "MDNsonKGiZQhdi4KXhxpm", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": -49.541731087778885, + "y": -2.504131122615604, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 59.83333206176758, + "height": 11.674843976770063, + "seed": 392707069, + "groupIds": [ + "wdHaa1NaRI3KXxcIXUuOw", + "XW1oTLv-UWuR0QNTPPlLb" + ], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 9.339875181416051, + "fontFamily": 1, + "text": "pod-service-a", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "pod-service-a", + "lineHeight": 1.25, + "baseline": 8, + "index": "b0A", + "frameId": null, + "autoResize": true + }, + { + "type": "ellipse", + "version": 492, + "versionNonce": 719480020, + "isDeleted": false, + "id": "Pr8_Sj1kfY3e6lcPsFAwO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 17.494800580705032, + "y": -22.36280906966772, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 36, + "height": 42, + "seed": 943282739, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "yUcyGWcG1vpkIQYphdkJF" + } + ], + "updated": 1737044619931, + "link": null, + "locked": false, + "index": "b0B", + "frameId": null + }, + { + "type": "text", + "version": 626, + "versionNonce": 914489964, + "isDeleted": false, + "id": "yUcyGWcG1vpkIQYphdkJF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 29.566878710082037, + "y": -11.212051474585223, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 11.399999618530273, + "height": 20, + "seed": 118856659, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "2", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "Pr8_Sj1kfY3e6lcPsFAwO", + "originalText": "2", + "lineHeight": 1.25, + "baseline": 14, + "index": "b0C", + "frameId": null, + "autoResize": true + }, + { + "id": "t2HmxF7YpW-wgtQBlwKTg", + "type": "image", + "x": 1584.5296771466633, + "y": -18.45577708511553, + "width": 67.00000000000001, + "height": 58, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "roundness": null, + "seed": 1020501336, + "version": 199, + "versionNonce": 1818667604, + "isDeleted": false, + "boundElements": [], + "updated": 1737044619931, + "link": null, + "locked": false, + "status": "saved", + "fileId": "77fdf834ff5deedcaf7ffeba1fbefe736638f24d", + "scale": [ + 1, + 1 + ], + "index": "b0D", + "frameId": null, + "crop": null + }, + { + "type": "line", + "version": 1303, + "versionNonce": 112015828, + "isDeleted": false, + "id": "g8NayyITe_62KXfzXpzN-", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1988.3868978091898, + "y": 33.359747919582446, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 49.26942071813747, + "height": 43.87300421060919, + "seed": 973240532, + "groupIds": [ + "jopepbrqZSL2EBO7USViC", + "jR0arybZMHY2Q-dMlmUMG" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1737044755152, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.518175120431392, + -29.072472669680792 + ], + [ + 23.649321944705985, + -43.87300421060919 + ], + [ + 41.780468768980576, + -32.244015142736835 + ], + [ + 49.26942071813747, + -3.0188078795298896 + ], + [ + 0, + 0 + ] + ], + "index": "b0F", + "frameId": null, + "roundness": { + "type": 2 + } + }, + { + "type": "ellipse", + "version": 989, + "versionNonce": 302964564, + "isDeleted": false, + "id": "V9SRLnEuUtnPSZ4D3tK9o", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 2000.8407686455055, + "y": -33.33129986627398, + "strokeColor": "#000000", + "backgroundColor": "#ced4da", + "width": 25.225943407686408, + "height": 22.072700481725683, + "seed": 1180974676, + "groupIds": [ + "jopepbrqZSL2EBO7USViC", + "jR0arybZMHY2Q-dMlmUMG" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1737044755152, + "link": null, + "locked": false, + "index": "b0G", + "frameId": null, + "roundness": null + } + ], + "appState": { + "gridSize": 20, + "gridStep": 5, + "gridModeEnabled": false, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "0644a8702c31871ef24e831da4f3ed873aa2d825": { + "mimeType": "image/png", + "id": "0644a8702c31871ef24e831da4f3ed873aa2d825", + "dataURL": "", + "created": 1726574587466, + "lastRetrieved": 1737043646801 + }, + "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78": { + "mimeType": "image/png", + "id": "06be7cf5d46d9cedaf2bf75a32d86c6d3b9f6c78", + "dataURL": "", + "created": 1726574587467, + "lastRetrieved": 1737043646801 + }, + "77fdf834ff5deedcaf7ffeba1fbefe736638f24d": { + "mimeType": "image/png", + "id": "77fdf834ff5deedcaf7ffeba1fbefe736638f24d", + "dataURL": "", + "created": 1686207156690, + "lastRetrieved": 1686207156690 + } + } +} \ No newline at end of file diff --git a/go.mod b/go.mod new file mode 100644 index 0000000..d66167e --- /dev/null +++ b/go.mod @@ -0,0 +1,99 @@ +module github.com/adevinta/observability-operator + +go 1.22.3 + +toolchain go1.23.2 + +require ( + github.com/adevinta/go-k8s-toolkit v0.0.0-20240913131011-109953036918 + github.com/adevinta/go-log-toolkit v0.0.0-20241010115832-de31e6161a8e + github.com/go-logr/logr v1.4.2 + github.com/grafana/grafana-com-public-clients/go/gcom v0.0.0-20241023082751-1bde2051eadd + github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring v0.50.0 + github.com/prometheus/client_golang v1.19.1 + github.com/prometheus/common v0.55.0 + github.com/prometheus/prometheus v1.8.2-0.20210701133801-b0944590a1c9 + github.com/sirupsen/logrus v1.9.3 + github.com/stretchr/testify v1.9.0 + gopkg.in/yaml.v3 v3.0.1 + k8s.io/api v0.31.1 + k8s.io/apimachinery v0.31.1 + k8s.io/autoscaler/vertical-pod-autoscaler v0.10.1-0.20240925135200-1c35e1186735 + k8s.io/client-go v0.31.1 + k8s.io/kubectl v0.31.0 + k8s.io/utils v0.0.0-20240711033017-18e509b52bc8 + sigs.k8s.io/controller-runtime v0.19.0 + sigs.k8s.io/e2e-framework v0.5.0 +) + +require ( + github.com/Microsoft/go-winio v0.5.0 // indirect + github.com/adevinta/go-system-toolkit v0.0.0-20240912143443-133d8c380cfc // indirect + github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 // indirect + github.com/armon/go-metrics v0.3.6 // indirect + github.com/aws/aws-sdk-go v1.44.11 // indirect + github.com/beorn7/perks v1.0.1 // indirect + github.com/blang/semver/v4 v4.0.0 // indirect + github.com/cespare/xxhash/v2 v2.3.0 // indirect + github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect + github.com/emicklei/go-restful/v3 v3.11.0 // indirect + github.com/evanphx/json-patch/v5 v5.9.0 // indirect + github.com/fsnotify/fsnotify v1.7.0 // indirect + github.com/fxamacker/cbor/v2 v2.7.0 // indirect + github.com/go-kit/log v0.2.1 // indirect + github.com/go-logfmt/logfmt v0.6.0 // indirect + github.com/go-openapi/jsonpointer v0.19.6 // indirect + github.com/go-openapi/jsonreference v0.20.2 // indirect + github.com/go-openapi/swag v0.22.4 // indirect + github.com/gogo/protobuf v1.3.2 // indirect + github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect + github.com/golang/protobuf v1.5.4 // indirect + github.com/google/gnostic-models v0.6.8 // indirect + github.com/google/go-cmp v0.6.0 // indirect + github.com/google/go-querystring v1.1.0 // indirect + github.com/google/gofuzz v1.2.0 // indirect + github.com/google/uuid v1.6.0 // indirect + github.com/gorilla/websocket v1.5.0 // indirect + github.com/imdario/mergo v0.3.15 // indirect + github.com/josharian/intern v1.0.0 // indirect + github.com/jpillora/backoff v1.0.0 // indirect + github.com/json-iterator/go v1.1.12 // indirect + github.com/mailru/easyjson v0.7.7 // indirect + github.com/mattn/go-colorable v0.1.13 // indirect + github.com/mattn/go-isatty v0.0.19 // indirect + github.com/miekg/dns v1.1.56 // indirect + github.com/moby/spdystream v0.4.0 // indirect + github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect + github.com/modern-go/reflect2 v1.0.2 // indirect + github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect + github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f // indirect + github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f // indirect + github.com/pkg/errors v0.9.1 // indirect + github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect + github.com/prometheus/client_model v0.6.1 // indirect + github.com/prometheus/procfs v0.15.1 // indirect + github.com/spf13/afero v1.8.2 // indirect + github.com/spf13/pflag v1.0.5 // indirect + github.com/vladimirvivien/gexe v0.3.0 // indirect + github.com/x448/float16 v0.8.4 // indirect + golang.org/x/exp v0.0.0-20230515195305-f3d0a9c9a5cc // indirect + golang.org/x/net v0.26.0 // indirect + golang.org/x/oauth2 v0.21.0 // indirect + golang.org/x/sys v0.21.0 // indirect + golang.org/x/term v0.21.0 // indirect + golang.org/x/text v0.16.0 // indirect + golang.org/x/time v0.4.0 // indirect + gomodules.xyz/jsonpatch/v2 v2.4.0 // indirect + google.golang.org/protobuf v1.34.2 // indirect + gopkg.in/evanphx/json-patch.v4 v4.12.0 // indirect + gopkg.in/inf.v0 v0.9.1 // indirect + gopkg.in/validator.v2 v2.0.1 // indirect + gopkg.in/yaml.v2 v2.4.0 // indirect + k8s.io/apiextensions-apiserver v0.31.0 // indirect + k8s.io/component-base v0.31.1 // indirect + k8s.io/klog/v2 v2.130.1 // indirect + k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 // indirect + sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect + sigs.k8s.io/structured-merge-diff/v4 v4.4.1 // indirect + sigs.k8s.io/yaml v1.4.0 // indirect +) diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..5be9884 --- /dev/null +++ b/go.sum @@ -0,0 +1,1709 @@ +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.43.0/go.mod h1:BOSR3VbTLkk6FDC/TcffxP4NF/FFBGA5ku+jvKOP7pg= +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.44.3/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.50.0/go.mod h1:r9sluTvynVuxRIOHXQEHMFffphuXHOMZMycpNR5e6To= +cloud.google.com/go v0.51.0/go.mod h1:hWtGJ6gnXH+KgDv+V0zFGDvpi07n3z8ZNj3T1RW0Gcw= +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.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKVk= +cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= +cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc= +cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY= +cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI= +cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk= +cloud.google.com/go v0.75.0/go.mod h1:VGuuCn7PG0dwsd5XPVm2Mm3wlh3EL55/79EKB6hlPTY= +cloud.google.com/go v0.78.0/go.mod h1:QjdrLG0uq+YwhjoVOLsS1t7TW8fs36kLs4XO5R5ECHg= +cloud.google.com/go v0.79.0/go.mod h1:3bzgcEeQlzbuEAYu4mrWhKqWjmpprinYgKJLgKHnbb8= +cloud.google.com/go v0.81.0/go.mod h1:mk/AM35KwGk/Nm2YSeZbxXdrNK3KZOYHmLkOqC2V6E0= +cloud.google.com/go v0.83.0/go.mod h1:Z7MJUsANfY0pYPdw0lbnivPx4/vhy/e2FEkSkF7vAVY= +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/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg= +cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc= +cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ= +cloud.google.com/go/bigtable v1.2.0/go.mod h1:JcVAOl45lrTmQfLj7T6TxyMzIN/3FGGcFm+2xVAli2o= +cloud.google.com/go/bigtable v1.3.0/go.mod h1:z5EyKrPE8OQmeg4h5MNdKvuSnI9CCT49Ki3f23aBzio= +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/pubsub v1.3.1/go.mod h1:i+ucay31+CNRpDW4Lu78I4xXG+O1r/MAHgjpRVR+TSU= +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= +cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs= +cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0= +cloud.google.com/go/storage v1.14.0/go.mod h1:GrKmX003DSIwi9o29oFT7YDnHYwZoctc3fOKtUw0Xmo= +collectd.org v0.3.0/go.mod h1:A/8DzQBkF6abtvrT2j/AU/4tiBgJWYyh0y/oB/4MlWE= +dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= +github.com/Azure/azure-sdk-for-go v41.3.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc= +github.com/Azure/azure-sdk-for-go v55.2.0+incompatible h1:TL2/vJWJEPOrmv97nHcbvjXES0Ntlb9P95hqGA1J2dU= +github.com/Azure/azure-sdk-for-go v55.2.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc= +github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8= +github.com/Azure/go-autorest v14.2.0+incompatible h1:V5VMDjClD3GiElqLWO7mz2MxNAK/vTfRHdAubSIPRgs= +github.com/Azure/go-autorest v14.2.0+incompatible/go.mod h1:r+4oMnoxhatjLLJ6zxSWATqVooLgysK6ZNox3g/xq24= +github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI= +github.com/Azure/go-autorest/autorest v0.9.3/go.mod h1:GsRuLYvwzLjjjRoWEIyMUaYq8GNUx2nRB378IPt/1p0= +github.com/Azure/go-autorest/autorest v0.10.0/go.mod h1:/FALq9T/kS7b5J5qsQ+RSTUdAmGFqi0vUdVNNx8q630= +github.com/Azure/go-autorest/autorest v0.10.1/go.mod h1:/FALq9T/kS7b5J5qsQ+RSTUdAmGFqi0vUdVNNx8q630= +github.com/Azure/go-autorest/autorest v0.11.12/go.mod h1:eipySxLmqSyC5s5k1CLupqet0PSENBEDP93LQ9a8QYw= +github.com/Azure/go-autorest/autorest v0.11.19 h1:7/IqD2fEYVha1EPeaiytVKhzmPV223pfkRIQUGOK2IE= +github.com/Azure/go-autorest/autorest v0.11.19/go.mod h1:dSiJPy22c3u0OtOKDNttNgqpNFY/GeWa7GH/Pz56QRA= +github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0= +github.com/Azure/go-autorest/autorest/adal v0.8.0/go.mod h1:Z6vX6WXXuyieHAXwMj0S6HY6e6wcHn37qQMBQlvY3lc= +github.com/Azure/go-autorest/autorest/adal v0.8.1/go.mod h1:ZjhuQClTqx435SRJ2iMlOxPYt3d2C/T/7TiQCVZSn3Q= +github.com/Azure/go-autorest/autorest/adal v0.8.2/go.mod h1:ZjhuQClTqx435SRJ2iMlOxPYt3d2C/T/7TiQCVZSn3Q= +github.com/Azure/go-autorest/autorest/adal v0.8.3/go.mod h1:ZjhuQClTqx435SRJ2iMlOxPYt3d2C/T/7TiQCVZSn3Q= +github.com/Azure/go-autorest/autorest/adal v0.9.5/go.mod h1:B7KF7jKIeC9Mct5spmyCB/A8CG/sEz1vwIRGv/bbw7A= +github.com/Azure/go-autorest/autorest/adal v0.9.13/go.mod h1:W/MM4U6nLxnIskrw4UwWzlHfGjwUS50aOsc/I3yuU8M= +github.com/Azure/go-autorest/autorest/adal v0.9.14 h1:G8hexQdV5D4khOXrWG2YuLCFKhWYmWD8bHYaXN5ophk= +github.com/Azure/go-autorest/autorest/adal v0.9.14/go.mod h1:W/MM4U6nLxnIskrw4UwWzlHfGjwUS50aOsc/I3yuU8M= +github.com/Azure/go-autorest/autorest/azure/auth v0.4.2/go.mod h1:90gmfKdlmKgfjUpnCEpOJzsUEjrWDSLwHIG73tSXddM= +github.com/Azure/go-autorest/autorest/azure/cli v0.3.1/go.mod h1:ZG5p860J94/0kI9mNJVoIoLgXcirM2gF5i2kWloofxw= +github.com/Azure/go-autorest/autorest/date v0.1.0/go.mod h1:plvfp3oPSKwf2DNjlBjWF/7vwR+cUD/ELuzDCXwHUVA= +github.com/Azure/go-autorest/autorest/date v0.2.0/go.mod h1:vcORJHLJEh643/Ioh9+vPmf1Ij9AEBM5FuBIXLmIy0g= +github.com/Azure/go-autorest/autorest/date v0.3.0 h1:7gUk1U5M/CQbp9WoqinNzJar+8KY+LPI6wiWrP/myHw= +github.com/Azure/go-autorest/autorest/date v0.3.0/go.mod h1:BI0uouVdmngYNUzGWeSYnokU+TrmwEsOqdt8Y6sso74= +github.com/Azure/go-autorest/autorest/mocks v0.1.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0= +github.com/Azure/go-autorest/autorest/mocks v0.2.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0= +github.com/Azure/go-autorest/autorest/mocks v0.3.0/go.mod h1:a8FDP3DYzQ4RYfVAxAN3SVSiiO77gL2j2ronKKP0syM= +github.com/Azure/go-autorest/autorest/mocks v0.4.1/go.mod h1:LTp+uSrOhSkaKrUy935gNZuuIPPVsHlr9DSOxSayd+k= +github.com/Azure/go-autorest/autorest/to v0.3.0/go.mod h1:MgwOyqaIuKdG4TL/2ywSsIWKAfJfgHDo8ObuUk3t5sA= +github.com/Azure/go-autorest/autorest/to v0.4.0 h1:oXVqrxakqqV1UZdSazDOPOLvOIz+XA683u8EctwboHk= +github.com/Azure/go-autorest/autorest/to v0.4.0/go.mod h1:fE8iZBn7LQR7zH/9XU2NcPR4o9jEImooCeWJcYV/zLE= +github.com/Azure/go-autorest/autorest/validation v0.2.0/go.mod h1:3EEqHnBxQGHXRYq3HT1WyXAvT7LLY3tl70hw6tQIbjI= +github.com/Azure/go-autorest/autorest/validation v0.3.1 h1:AgyqjAd94fwNAoTjl/WQXg4VvFeRFpO+UhNyRXqF1ac= +github.com/Azure/go-autorest/autorest/validation v0.3.1/go.mod h1:yhLgjC0Wda5DYXl6JAsWyUe4KVNffhoDhG0zVzUMo3E= +github.com/Azure/go-autorest/logger v0.1.0/go.mod h1:oExouG+K6PryycPJfVSxi/koC6LSNgds39diKLz7Vrc= +github.com/Azure/go-autorest/logger v0.2.0/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8= +github.com/Azure/go-autorest/logger v0.2.1 h1:IG7i4p/mDa2Ce4TRyAO8IHnVhAVF3RFU+ZtXWSmf4Tg= +github.com/Azure/go-autorest/logger v0.2.1/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8= +github.com/Azure/go-autorest/tracing v0.5.0/go.mod h1:r/s2XiOKccPW3HrqB+W0TQzfbtp2fGCgRFtBroKn4Dk= +github.com/Azure/go-autorest/tracing v0.6.0 h1:TYi4+3m5t6K48TGI9AUdb+IzbnSxvnvUMfuitfgcfuo= +github.com/Azure/go-autorest/tracing v0.6.0/go.mod h1:+vhtPC754Xsa23ID7GlGsrdKBpUA79WCAKPPZVC2DeU= +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/DATA-DOG/go-sqlmock v1.4.1/go.mod h1:f/Ixk793poVmq4qj/V1dPUg2JEAKC73Q5eFN3EC/SaM= +github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= +github.com/HdrHistogram/hdrhistogram-go v1.0.1/go.mod h1:BWJ+nMSHY3L41Zj7CA3uXnloDp7xxV0YvstAE7nKTaM= +github.com/Knetic/govaluate v3.0.1-0.20171022003610-9aa49832a739+incompatible/go.mod h1:r7JcOSlj0wfOMncg0iLm8Leh48TZaKVeNIfJntJ2wa0= +github.com/Masterminds/semver v1.4.2/go.mod h1:MB6lktGJrhw8PrUyiEoblNEGEQ+RzHPF078ddwwvV3Y= +github.com/Masterminds/sprig v2.16.0+incompatible/go.mod h1:y6hNFY5UBTIWBxnzTeuNhlNS5hqE0NB0E6fgfo2Br3o= +github.com/Microsoft/go-winio v0.4.16/go.mod h1:XB6nPKklQyQ7GC9LdcBEcBl8PF76WugXOPRXwdLnMv0= +github.com/Microsoft/go-winio v0.5.0 h1:Elr9Wn+sGKPlkaBvwu4mTrxtmOp3F3yV9qhaHbXGjwU= +github.com/Microsoft/go-winio v0.5.0/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= +github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ= +github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= +github.com/PuerkitoBio/purell v1.0.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= +github.com/PuerkitoBio/purell v1.1.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= +github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0= +github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= +github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE= +github.com/SAP/go-hdb v0.14.1/go.mod h1:7fdQLVC2lER3urZLjZCm0AuMQfApof92n3aylBPEkMo= +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/adevinta/go-k8s-toolkit v0.0.0-20240913131011-109953036918 h1:/Ikp6EAC4VlvvT7621McUUXHeebMkp3idCd2wOkcTmA= +github.com/adevinta/go-k8s-toolkit v0.0.0-20240913131011-109953036918/go.mod h1:VQZFxcRYsH+GGOsn3pTpIfW91l1NxxwYpo7MHLC/w7A= +github.com/adevinta/go-log-toolkit v0.0.0-20241010115832-de31e6161a8e h1:jakT5Q3QRQ76OdkvzZViZMQMvSxnqoJoC3V2tngdv90= +github.com/adevinta/go-log-toolkit v0.0.0-20241010115832-de31e6161a8e/go.mod h1:oKcFLGHSHtWJRos+gkZ/tji5U4v6f+f8DMRP+91G+Yg= +github.com/adevinta/go-system-toolkit v0.0.0-20240912143443-133d8c380cfc h1:AjWBRPpsZRIubsXViIgTPdGRF1qPTGMg5/zKzfu7xgQ= +github.com/adevinta/go-system-toolkit v0.0.0-20240912143443-133d8c380cfc/go.mod h1:67b2+kw34iCi66dfDCZvug3Dm+yKyAfYKMC9ahdv35I= +github.com/adevinta/go-testutils-toolkit v0.0.0-20240913074508-af35ec32d0a7 h1:x8emhWKZ9Y6TO6750MOQU7BlHtCsjmfkp4AV5MwzPpU= +github.com/adevinta/go-testutils-toolkit v0.0.0-20240913074508-af35ec32d0a7/go.mod h1:ylsX6yclqK6PscEATGV8jzAzanx2CxeX7ycjEhk/KRE= +github.com/afex/hystrix-go v0.0.0-20180502004556-fa1af6a1f4f5/go.mod h1:SkGFH1ia65gfNATL8TAiHDNxPzPdmEL5uirI2Uyuz6c= +github.com/agnivade/levenshtein v1.0.1/go.mod h1:CURSv5d9Uaml+FovSIICkLbAUZ9S4RqaHDIsdSBg7lM= +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/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= +github.com/alecthomas/units v0.0.0-20210208195552-ff826a37aa15/go.mod h1:OMCwj8VM1Kc9e19TLln2VL61YJF0x1XFtfdL4JdbSyE= +github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 h1:s6gZFSlWYmbqAuRjVTiNNhvNRfY2Wxp9nhfyel4rklc= +github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137/go.mod h1:OMCwj8VM1Kc9e19TLln2VL61YJF0x1XFtfdL4JdbSyE= +github.com/andreyvit/diff v0.0.0-20170406064948-c7f18ee00883/go.mod h1:rCTlJbsFo29Kk6CurOXKm700vrz8f0KW0JNfpkRJY/8= +github.com/antihax/optional v0.0.0-20180407024304-ca021399b1a6/go.mod h1:V8iCPQYkqmusNa815XgQio277wI47sdRh1dUOLdyC6Q= +github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= +github.com/aokoli/goutils v1.0.1/go.mod h1:SijmP0QR8LtwsmDs8Yii5Z/S4trXFGFC2oO5g9DP+DQ= +github.com/apache/arrow/go/arrow v0.0.0-20191024131854-af6fa24be0db/go.mod h1:VTxUBvSJ3s3eHAg65PNgrsn5BtqCRPdmyXh6rAfdxN0= +github.com/apache/arrow/go/arrow v0.0.0-20200923215132-ac86123a3f01/go.mod h1:QNYViu/X0HXDHw7m3KXzWSVXIbfUvJqBFe6Gj8/pYA0= +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= +github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= +github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= +github.com/armon/go-metrics v0.3.3/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= +github.com/armon/go-metrics v0.3.6 h1:x/tmtOF9cDBoXH7XoAGOz2qqm1DknFD1590XmD/DUJ8= +github.com/armon/go-metrics v0.3.6/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= +github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= +github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= +github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5 h1:0CwZNZbxp69SHPdPJAN/hZIm0C4OItdklCFmMRWYpio= +github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs= +github.com/aryann/difflib v0.0.0-20170710044230-e206f873d14a/go.mod h1:DAHtR1m6lCRdSC2Tm3DSWRPvIPr6xNKyeHdqDQSQT+A= +github.com/asaskevich/govalidator v0.0.0-20180720115003-f9ffefc3facf/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY= +github.com/asaskevich/govalidator v0.0.0-20190424111038-f61b66f89f4a/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY= +github.com/asaskevich/govalidator v0.0.0-20200108200545-475eaeb16496/go.mod h1:oGkLhpf+kjZl6xBf758TQhh5XrAeiJv/7FRz/2spLIg= +github.com/asaskevich/govalidator v0.0.0-20200428143746-21a406dcc535/go.mod h1:oGkLhpf+kjZl6xBf758TQhh5XrAeiJv/7FRz/2spLIg= +github.com/asaskevich/govalidator v0.0.0-20200907205600-7a23bdc65eef/go.mod h1:WaHUgvxTVq04UNunO+XhnAqY/wQc+bxr74GqbsZ/Jqw= +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.29.16/go.mod h1:1KvfttTE3SPKMpo8g2c6jL3ZKfXtFvKscTgahTma5Xg= +github.com/aws/aws-sdk-go v1.30.12/go.mod h1:5zCpMtNQVjRREroY7sYe8lOMRSxkhG6MZveU8YkpAk0= +github.com/aws/aws-sdk-go v1.34.28/go.mod h1:H7NKnBqNVzoTJpGfLrQkkD+ytBA93eiDYi/+8rV9s48= +github.com/aws/aws-sdk-go v1.38.60/go.mod h1:hcU610XS61/+aQV88ixoOzUoG7v3b31pl2zKMmprdro= +github.com/aws/aws-sdk-go v1.44.11 h1:eIC59RrNY7xXYmGy/kKkLj4PGB325Jca22lcxZwbpBE= +github.com/aws/aws-sdk-go v1.44.11/go.mod h1:y4AeaBuwd2Lk+GepC1E9v0qOiTws0MIWAX4oIKwKHZo= +github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= +github.com/benbjohnson/immutable v0.2.1/go.mod h1:uc6OHo6PN2++n98KHLxW8ef4W42ylHiQSENghE1ezxI= +github.com/benbjohnson/tmpl v1.0.0/go.mod h1:igT620JFIi44B6awvU9IsDhR77IXWtFigTLil/RPdps= +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 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= +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/blang/semver v3.5.0+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk= +github.com/blang/semver/v4 v4.0.0 h1:1PFHFE6yCCTv8C1TeyNNarDzntLi7wMI5i/pzqYIsAM= +github.com/blang/semver/v4 v4.0.0/go.mod h1:IbckMUScFkM3pff0VJDNKRiT6TG/YpiHIM2yvyW5YoQ= +github.com/bmizerany/pat v0.0.0-20170815010413-6226ea591a40/go.mod h1:8rLXio+WjiTceGBHIoTvn60HIbs7Hm7bcHjyrSqYB9c= +github.com/boltdb/bolt v1.3.1/go.mod h1:clJnj/oiGkjum5o1McbSZDSLxVThjynRyGBgiAx27Ps= +github.com/bonitoo-io/go-sql-bigquery v0.3.4-1.4.0/go.mod h1:J4Y6YJm0qTWB9aFziB7cPeSyc6dOZFyJdteSeybVpXQ= +github.com/c-bata/go-prompt v0.2.2/go.mod h1:VzqtzE2ksDBcdln8G7mk2RX9QyGjH+OVqOCSiVIqS34= +github.com/cactus/go-statsd-client/statsd v0.0.0-20191106001114-12b4e2b38748/go.mod h1:l/bIBLeOl9eX+wxJAzxS4TveKRtAqlyDpHjhkfO0MEI= +github.com/casbin/casbin/v2 v2.1.2/go.mod h1:YcPU1XXisHhLzuxH9coDNf2FbKpjGlbCg3n9yuLkIJQ= +github.com/cenkalti/backoff v0.0.0-20181003080854-62661b46c409/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= +github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= +github.com/cenkalti/backoff/v4 v4.1.0/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= +github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= +github.com/cespare/xxhash/v2 v2.1.0/go.mod h1:dgIUBU3pDso/gPgZ1osOZ0iQf77oPR28Tjxl5dIMyVM= +github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= +github.com/cespare/xxhash/v2 v2.3.0/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/circonus-labs/circonus-gometrics v2.3.1+incompatible/go.mod h1:nmEj6Dob7S7YxXgwXpfOuvO54S+tGdZdw9fuRZt25Ag= +github.com/circonus-labs/circonusllhist v0.1.3/go.mod h1:kMXHVDlOchFAehlya5ePtbp5jckzBHf4XRpQvBOLI+I= +github.com/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/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/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/containerd/containerd v1.4.3 h1:ijQT13JedHSHrQGWFcGEwzcNKrAGIiZ+jSD5QQG07SY= +github.com/containerd/containerd v1.4.3/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA= +github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= +github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= +github.com/coreos/go-oidc v2.1.0+incompatible/go.mod h1:CgnwVTmzoESiwO9qyAFEMiHoZ1nMCKZlZ9V6mm3/LKc= +github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-semver v0.3.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-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= +github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= +github.com/coreos/pkg v0.0.0-20180108230652-97fdf19511ea/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= +github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= +github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/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/creack/pty v1.1.11/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= +github.com/dave/jennifer v1.2.0/go.mod h1:fIb+770HOpJ2fmN9EPPKOqm1vMGhB+TwXKMZhrIygKg= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= +github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/denisenkom/go-mssqldb v0.0.0-20200428022330-06a60b6afbbc/go.mod h1:xbL0rPBG9cCiLr28tMa8zpbdarY27NDyej4t/EjAShU= +github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= +github.com/dgrijalva/jwt-go/v4 v4.0.0-preview1/go.mod h1:+hnT3ywWDTAFrW5aE+u2Sa/wT555ZqwoCS+pk3p6ry4= +github.com/dgryski/go-bitstream v0.0.0-20180413035011-3522498ce2c8/go.mod h1:VMaSuZ+SZcx/wljOQKvp5srsbCiKDEb6K2wC4+PiBmQ= +github.com/dgryski/go-sip13 v0.0.0-20190329191031-25c5027a8c7b/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= +github.com/dgryski/go-sip13 v0.0.0-20200911182023-62edffca9245/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= +github.com/digitalocean/godo v1.62.0 h1:7Gw2KFsWkxl36qJa0s50tgXaE0Cgm51JdRP+MFQvNnM= +github.com/digitalocean/godo v1.62.0/go.mod h1:p7dOjjtSBqCTUksqtA5Fd3uaKs9kyTq2xcz76ulEJRU= +github.com/dimchansky/utfbom v1.1.0/go.mod h1:rO41eb7gLfo8SF1jd9F8HplJm1Fewwi4mQvIirEdv+8= +github.com/dnaeon/go-vcr v1.0.1/go.mod h1:aBB1+wY4s93YsC3HHjMBMrwTj2R9FHDzUr9KyGc8n1E= +github.com/docker/distribution v2.7.1+incompatible h1:a5mlkVzth6W5A4fOsS3D2EO5BUmsJpcB+cRlLU7cSug= +github.com/docker/distribution v2.7.1+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= +github.com/docker/docker v0.7.3-0.20190327010347-be7ac8be2ae0/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= +github.com/docker/docker v20.10.7+incompatible h1:Z6O9Nhsjv+ayUEeI1IojKbYcsGdgYSNqxe1s2MYzUhQ= +github.com/docker/docker v20.10.7+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= +github.com/docker/go-connections v0.4.0 h1:El9xVISelRB7BuFusrZozjnkIM5YnzCViNKohAFqRJQ= +github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec= +github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= +github.com/docker/go-units v0.4.0 h1:3uh0PgVws3nIA0Q+MwDC8yjEPf9zjRfZZWXZYDct3Tw= +github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= +github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM= +github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE= +github.com/dustin/go-humanize v0.0.0-20171111073723-bb3d318650d4/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= +github.com/dustin/go-humanize v1.0.0/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= +github.com/eapache/queue v1.1.0/go.mod h1:6eCeP0CKFpHLu8blIFXhExK/dRa7WDZfr6jVFPTqq+I= +github.com/eclipse/paho.mqtt.golang v1.2.0/go.mod h1:H9keYFcgq3Qr5OUJm/JZI/i6U7joQ8SYLhZwfeOo6Ts= +github.com/edsrzf/mmap-go v1.0.0/go.mod h1:YO35OhQPt3KJa3ryjFM5Bs14WD66h8eGKpfaBNrHW5M= +github.com/elazarl/goproxy v0.0.0-20170405201442-c4fc26588b6e/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc= +github.com/elazarl/goproxy v0.0.0-20180725130230-947c36da3153/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc= +github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs= +github.com/emicklei/go-restful v2.9.5+incompatible/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs= +github.com/emicklei/go-restful/v3 v3.11.0 h1:rAQeMHw1c7zTmncogyy8VvRZwtkmkZ4FxERmMY4rD+g= +github.com/emicklei/go-restful/v3 v3.11.0/go.mod h1:6n3XBCmQQb25CM2LCACGz8ukIrRry+4bhvbpWn3mrbc= +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/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po= +github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= +github.com/envoyproxy/go-control-plane v0.9.9-0.20210217033140-668b12f5399d/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= +github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= +github.com/evanphx/json-patch v4.2.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= +github.com/evanphx/json-patch v4.9.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= +github.com/evanphx/json-patch v4.12.0+incompatible h1:4onqiflcdA9EOZ4RxV643DvftH5pOlLGNtQ5lPWQu84= +github.com/evanphx/json-patch v4.12.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= +github.com/evanphx/json-patch/v5 v5.9.0 h1:kcBlZQbplgElYIlo/n1hJbls2z/1awpXxpRi0/FOJfg= +github.com/evanphx/json-patch/v5 v5.9.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ= +github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= +github.com/fatih/color v1.9.0 h1:8xPHl4/q1VyqGIPif1F+1V3Y3lSmrq01EabUW3CoW5s= +github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= +github.com/fogleman/gg v1.2.1-0.20190220221249-0403632d5b90/go.mod h1:R/bRT+9gY/C5z7JzPU0zXsXHKM4/ayA+zqcVNZzPa1k= +github.com/form3tech-oss/jwt-go v3.2.2+incompatible h1:TcekIExNqud5crz4xD2pavyTgWiPvpYe4Xau31I0PRk= +github.com/form3tech-oss/jwt-go v3.2.2+incompatible/go.mod h1:pbq4aXjuKjdthFRnoDwaVPLA+WlJuPGy+QneDUgJi2k= +github.com/foxcpp/go-mockdns v0.0.0-20201212160233-ede2f9158d15/go.mod h1:tPg4cp4nseejPd+UKxtCVQ2hUxNTZ7qQZJa7CLriIeo= +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/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= +github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA= +github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= +github.com/fxamacker/cbor/v2 v2.7.0 h1:iM5WgngdRBanHcxugY4JySA0nk1wZorNOpTgCMedv5E= +github.com/fxamacker/cbor/v2 v2.7.0/go.mod h1:pxXPTn3joSm21Gbwsv0w9OSA2y1HFR9qXEeXQVeNoDQ= +github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/globalsign/mgo v0.0.0-20180905125535-1ca0a4f7cbcb/go.mod h1:xkRDCp4j0OGD1HRkm4kmhM+pmpv3AKq5SU7GMg4oO/Q= +github.com/globalsign/mgo v0.0.0-20181015135952-eeefdecb41b8/go.mod h1:xkRDCp4j0OGD1HRkm4kmhM+pmpv3AKq5SU7GMg4oO/Q= +github.com/glycerine/go-unsnap-stream v0.0.0-20180323001048-9f0cb55181dd/go.mod h1:/20jfyN9Y5QPEAprSgKAUr+glWDY39ZiUEAYOEv5dsE= +github.com/glycerine/goconvey v0.0.0-20190410193231-58a59202ab31/go.mod h1:Ogl1Tioa0aV7gstGFO7KhffUsb9M4ydbEbbxpcEDc24= +github.com/go-chi/chi v4.1.0+incompatible/go.mod h1:eB3wogJHnLi3x/kFX2A+IbTBlXxmMeXJVKy9tTv1XzQ= +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= +github.com/go-kit/kit v0.10.0/go.mod h1:xUsJbQ/Fp4kEt7AFgCuvyX4a71u8h9jB8tj/ORgOZ7o= +github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= +github.com/go-kit/log v0.2.1 h1:MRVx0/zhvdseW+Gza6N9rVzU/IVzaeE1SFI4raAhmBU= +github.com/go-kit/log v0.2.1/go.mod h1:NwTd00d/i8cPZ3xOwwiv2PO5MOcx78fFErGNcVmBjv0= +github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= +github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= +github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= +github.com/go-logfmt/logfmt v0.6.0 h1:wGYYu3uicYdqXVgoYbvnkrPVXkuLM1p1ifugDMEdRi4= +github.com/go-logfmt/logfmt v0.6.0/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= +github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas= +github.com/go-logr/logr v0.4.0/go.mod h1:z6/tIYblkpsD+a4lm/fGIIU9mZ+XfAiaFtq7xTgseGU= +github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY= +github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= +github.com/go-logr/zapr v1.3.0 h1:XGdV8XW8zdwFiwOA2Dryh1gj2KRQyOOoNmBy4EplIcQ= +github.com/go-logr/zapr v1.3.0/go.mod h1:YKepepNBd1u/oyhd/yQmtjVXmm9uML4IXUgMOwR8/Gg= +github.com/go-openapi/analysis v0.0.0-20180825180245-b006789cd277/go.mod h1:k70tL6pCuVxPJOHXQ+wIac1FUrvNkHolPie/cLEU6hI= +github.com/go-openapi/analysis v0.17.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik= +github.com/go-openapi/analysis v0.18.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik= +github.com/go-openapi/analysis v0.19.2/go.mod h1:3P1osvZa9jKjb8ed2TPng3f0i/UY9snX6gxi44djMjk= +github.com/go-openapi/analysis v0.19.4/go.mod h1:3P1osvZa9jKjb8ed2TPng3f0i/UY9snX6gxi44djMjk= +github.com/go-openapi/analysis v0.19.5/go.mod h1:hkEAkxagaIvIP7VTn8ygJNkd4kAYON2rCu0v0ObL0AU= +github.com/go-openapi/analysis v0.19.10/go.mod h1:qmhS3VNFxBlquFJ0RGoDtylO9y4pgTAUNE9AEEMdlJQ= +github.com/go-openapi/analysis v0.19.16/go.mod h1:GLInF007N83Ad3m8a/CbQ5TPzdnGT7workfHwuVjNVk= +github.com/go-openapi/analysis v0.20.0/go.mod h1:BMchjvaHDykmRMsK40iPtvyOfFdMMxlOmQr9FBZk+Og= +github.com/go-openapi/errors v0.17.0/go.mod h1:LcZQpmvG4wyF5j4IhA73wkLFQg+QJXOQHVjmcZxhka0= +github.com/go-openapi/errors v0.18.0/go.mod h1:LcZQpmvG4wyF5j4IhA73wkLFQg+QJXOQHVjmcZxhka0= +github.com/go-openapi/errors v0.19.2/go.mod h1:qX0BLWsyaKfvhluLejVpVNwNRdXZhEbTA4kxxpKBC94= +github.com/go-openapi/errors v0.19.3/go.mod h1:qX0BLWsyaKfvhluLejVpVNwNRdXZhEbTA4kxxpKBC94= +github.com/go-openapi/errors v0.19.4/go.mod h1:qX0BLWsyaKfvhluLejVpVNwNRdXZhEbTA4kxxpKBC94= +github.com/go-openapi/errors v0.19.6/go.mod h1:cM//ZKUKyO06HSwqAelJ5NsEMMcpa6VpXe8DOa1Mi1M= +github.com/go-openapi/errors v0.19.7/go.mod h1:cM//ZKUKyO06HSwqAelJ5NsEMMcpa6VpXe8DOa1Mi1M= +github.com/go-openapi/errors v0.19.8/go.mod h1:cM//ZKUKyO06HSwqAelJ5NsEMMcpa6VpXe8DOa1Mi1M= +github.com/go-openapi/errors v0.19.9/go.mod h1:cM//ZKUKyO06HSwqAelJ5NsEMMcpa6VpXe8DOa1Mi1M= +github.com/go-openapi/errors v0.20.0/go.mod h1:cM//ZKUKyO06HSwqAelJ5NsEMMcpa6VpXe8DOa1Mi1M= +github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0= +github.com/go-openapi/jsonpointer v0.17.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M= +github.com/go-openapi/jsonpointer v0.18.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M= +github.com/go-openapi/jsonpointer v0.19.2/go.mod h1:3akKfEdA7DF1sugOqz1dVQHBcuDBPKZGEoHC/NkiQRg= +github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= +github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= +github.com/go-openapi/jsonpointer v0.19.6 h1:eCs3fxoIi3Wh6vtgmLTOjdhSpiqphQ+DaPn38N2ZdrE= +github.com/go-openapi/jsonpointer v0.19.6/go.mod h1:osyAmYz/mB/C3I+WsTTSgw1ONzaLJoLCyoi6/zppojs= +github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg= +github.com/go-openapi/jsonreference v0.17.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I= +github.com/go-openapi/jsonreference v0.18.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I= +github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwohSTlpa0o73RUL1owJc= +github.com/go-openapi/jsonreference v0.19.3/go.mod h1:rjx6GuL8TTa9VaixXglHmQmIL98+wF9xc8zWvFonSJ8= +github.com/go-openapi/jsonreference v0.19.5/go.mod h1:RdybgQwPxbL4UEjuAruzK1x3nE69AqPYEJeo/TWfEeg= +github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2KvnJRumpMGbE= +github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k= +github.com/go-openapi/loads v0.17.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU= +github.com/go-openapi/loads v0.18.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU= +github.com/go-openapi/loads v0.19.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU= +github.com/go-openapi/loads v0.19.2/go.mod h1:QAskZPMX5V0C2gvfkGZzJlINuP7Hx/4+ix5jWFxsNPs= +github.com/go-openapi/loads v0.19.3/go.mod h1:YVfqhUCdahYwR3f3iiwQLhicVRvLlU/WO5WPaZvcvSI= +github.com/go-openapi/loads v0.19.4/go.mod h1:zZVHonKd8DXyxyw4yfnVjPzBjIQcLt0CCsn0N0ZrQsk= +github.com/go-openapi/loads v0.19.5/go.mod h1:dswLCAdonkRufe/gSUC3gN8nTSaB9uaS2es0x5/IbjY= +github.com/go-openapi/loads v0.19.6/go.mod h1:brCsvE6j8mnbmGBh103PT/QLHfbyDxA4hsKvYBNEGVc= +github.com/go-openapi/loads v0.19.7/go.mod h1:brCsvE6j8mnbmGBh103PT/QLHfbyDxA4hsKvYBNEGVc= +github.com/go-openapi/loads v0.20.0/go.mod h1:2LhKquiE513rN5xC6Aan6lYOSddlL8Mp20AW9kpviM4= +github.com/go-openapi/loads v0.20.2/go.mod h1:hTVUotJ+UonAMMZsvakEgmWKgtulweO9vYP2bQYKA/o= +github.com/go-openapi/runtime v0.0.0-20180920151709-4f900dc2ade9/go.mod h1:6v9a6LTXWQCdL8k1AO3cvqx5OtZY/Y9wKTgaoP6YRfA= +github.com/go-openapi/runtime v0.19.0/go.mod h1:OwNfisksmmaZse4+gpV3Ne9AyMOlP1lt4sK4FXt0O64= +github.com/go-openapi/runtime v0.19.4/go.mod h1:X277bwSUBxVlCYR3r7xgZZGKVvBd/29gLDlFGtJ8NL4= +github.com/go-openapi/runtime v0.19.15/go.mod h1:dhGWCTKRXlAfGnQG0ONViOZpjfg0m2gUt9nTQPQZuoo= +github.com/go-openapi/runtime v0.19.16/go.mod h1:5P9104EJgYcizotuXhEuUrzVc+j1RiSjahULvYmlv98= +github.com/go-openapi/runtime v0.19.24/go.mod h1:Lm9YGCeecBnUUkFTxPC4s1+lwrkJ0pthx8YvyjCfkgk= +github.com/go-openapi/runtime v0.19.28/go.mod h1:BvrQtn6iVb2QmiVXRsFAm6ZCAZBpbVKFfN6QWCp582M= +github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc= +github.com/go-openapi/spec v0.17.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI= +github.com/go-openapi/spec v0.18.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI= +github.com/go-openapi/spec v0.19.2/go.mod h1:sCxk3jxKgioEJikev4fgkNmwS+3kuYdJtcsZsD5zxMY= +github.com/go-openapi/spec v0.19.3/go.mod h1:FpwSN1ksY1eteniUU7X0N/BgJ7a4WvBFVA8Lj9mJglo= +github.com/go-openapi/spec v0.19.6/go.mod h1:Hm2Jr4jv8G1ciIAo+frC/Ft+rR2kQDh8JHKHb3gWUSk= +github.com/go-openapi/spec v0.19.7/go.mod h1:Hm2Jr4jv8G1ciIAo+frC/Ft+rR2kQDh8JHKHb3gWUSk= +github.com/go-openapi/spec v0.19.8/go.mod h1:Hm2Jr4jv8G1ciIAo+frC/Ft+rR2kQDh8JHKHb3gWUSk= +github.com/go-openapi/spec v0.19.15/go.mod h1:+81FIL1JwC5P3/Iuuozq3pPE9dXdIEGxFutcFKaVbmU= +github.com/go-openapi/spec v0.20.0/go.mod h1:+81FIL1JwC5P3/Iuuozq3pPE9dXdIEGxFutcFKaVbmU= +github.com/go-openapi/spec v0.20.1/go.mod h1:93x7oh+d+FQsmsieroS4cmR3u0p/ywH649a3qwC9OsQ= +github.com/go-openapi/spec v0.20.3/go.mod h1:gG4F8wdEDN+YPBMVnzE85Rbhf+Th2DTvA9nFPQ5AYEg= +github.com/go-openapi/strfmt v0.17.0/go.mod h1:P82hnJI0CXkErkXi8IKjPbNBM6lV6+5pLP5l494TcyU= +github.com/go-openapi/strfmt v0.18.0/go.mod h1:P82hnJI0CXkErkXi8IKjPbNBM6lV6+5pLP5l494TcyU= +github.com/go-openapi/strfmt v0.19.0/go.mod h1:+uW+93UVvGGq2qGaZxdDeJqSAqBqBdl+ZPMF/cC8nDY= +github.com/go-openapi/strfmt v0.19.2/go.mod h1:0yX7dbo8mKIvc3XSKp7MNfxw4JytCfCD6+bY1AVL9LU= +github.com/go-openapi/strfmt v0.19.3/go.mod h1:0yX7dbo8mKIvc3XSKp7MNfxw4JytCfCD6+bY1AVL9LU= +github.com/go-openapi/strfmt v0.19.4/go.mod h1:eftuHTlB/dI8Uq8JJOyRlieZf+WkkxUuk0dgdHXr2Qk= +github.com/go-openapi/strfmt v0.19.5/go.mod h1:eftuHTlB/dI8Uq8JJOyRlieZf+WkkxUuk0dgdHXr2Qk= +github.com/go-openapi/strfmt v0.19.11/go.mod h1:UukAYgTaQfqJuAFlNxxMWNvMYiwiXtLsF2VwmoFtbtc= +github.com/go-openapi/strfmt v0.20.0/go.mod h1:UukAYgTaQfqJuAFlNxxMWNvMYiwiXtLsF2VwmoFtbtc= +github.com/go-openapi/strfmt v0.20.1/go.mod h1:43urheQI9dNtE5lTZQfuFJvjYJKPrxicATpEfZwHUNk= +github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I= +github.com/go-openapi/swag v0.17.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg= +github.com/go-openapi/swag v0.18.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg= +github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= +github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= +github.com/go-openapi/swag v0.19.7/go.mod h1:ao+8BpOPyKdpQz3AOJfbeEVpLmWAvlT1IfTe5McPyhY= +github.com/go-openapi/swag v0.19.9/go.mod h1:ao+8BpOPyKdpQz3AOJfbeEVpLmWAvlT1IfTe5McPyhY= +github.com/go-openapi/swag v0.19.12/go.mod h1:eFdyEBkTdoAf/9RXBvj4cr1nH7GD8Kzo5HTt47gr72M= +github.com/go-openapi/swag v0.19.13/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ= +github.com/go-openapi/swag v0.19.14/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ= +github.com/go-openapi/swag v0.19.15/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ= +github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14= +github.com/go-openapi/swag v0.22.4 h1:QLMzNJnMGPRNDCbySlcj1x01tzU8/9LTTL9hZZZogBU= +github.com/go-openapi/swag v0.22.4/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14= +github.com/go-openapi/validate v0.18.0/go.mod h1:Uh4HdOzKt19xGIGm1qHf/ofbX1YQ4Y+MYsct2VUrAJ4= +github.com/go-openapi/validate v0.19.2/go.mod h1:1tRCw7m3jtI8eNWEEliiAqUIcBztB2KDnRCRMUi7GTA= +github.com/go-openapi/validate v0.19.3/go.mod h1:90Vh6jjkTn+OT1Eefm0ZixWNFjhtOH7vS9k0lo6zwJo= +github.com/go-openapi/validate v0.19.5/go.mod h1:8DJv2CVJQ6kGNpFW6eV9N3JviE1C85nY1c2z52x1Gk4= +github.com/go-openapi/validate v0.19.8/go.mod h1:8DJv2CVJQ6kGNpFW6eV9N3JviE1C85nY1c2z52x1Gk4= +github.com/go-openapi/validate v0.19.10/go.mod h1:RKEZTUWDkxKQxN2jDT7ZnZi2bhZlbNMAuKvKB+IaGx8= +github.com/go-openapi/validate v0.19.12/go.mod h1:Rzou8hA/CBw8donlS6WNEUQupNvUZ0waH08tGe6kAQ4= +github.com/go-openapi/validate v0.19.15/go.mod h1:tbn/fdOwYHgrhPBzidZfJC2MIVvs9GA7monOmWBbeCI= +github.com/go-openapi/validate v0.20.1/go.mod h1:b60iJT+xNNLfaQJUqLI7946tYiFEOuE9E4k54HpKcJ0= +github.com/go-openapi/validate v0.20.2/go.mod h1:e7OJoKNgd0twXZwIn0A43tHbvIcr/rZIVCbJBpTUoY0= +github.com/go-resty/resty/v2 v2.1.1-0.20191201195748-d7b97669fe48 h1:JVrqSeQfdhYRFk24TvhTZWU0q8lfCojxZQFi3Ou7+uY= +github.com/go-resty/resty/v2 v2.1.1-0.20191201195748-d7b97669fe48/go.mod h1:dZGr0i9PLlaaTD4H/hoZIDjQ+r6xq8mgbRzHZf7f2J8= +github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= +github.com/go-sql-driver/mysql v1.4.1/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/go-task/slim-sprig/v3 v3.0.0 h1:sUs3vkvUymDpBKi3qH1YSqBQk9+9D/8M2mN1vB6EwHI= +github.com/go-task/slim-sprig/v3 v3.0.0/go.mod h1:W848ghGpv3Qj3dhTPRyJypKRiqCdHZiAzKg9hl15HA8= +github.com/go-zookeeper/zk v1.0.2 h1:4mx0EYENAdX/B/rbunjlt5+4RTA/a9SMHBRuSKdGxPM= +github.com/go-zookeeper/zk v1.0.2/go.mod h1:nOB03cncLtlp4t+UAkGSV+9beXP/akpekBwL+UX1Qcw= +github.com/gobuffalo/attrs v0.0.0-20190224210810-a9411de4debd/go.mod h1:4duuawTqi2wkkpB4ePgWMaai6/Kc6WEz83bhFwpHzj0= +github.com/gobuffalo/depgen v0.0.0-20190329151759-d478694a28d3/go.mod h1:3STtPUQYuzV0gBVOY3vy6CfMm/ljR4pABfrTeHNLHUY= +github.com/gobuffalo/depgen v0.1.0/go.mod h1:+ifsuy7fhi15RWncXQQKjWS9JPkdah5sZvtHc2RXGlg= +github.com/gobuffalo/envy v1.6.15/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI= +github.com/gobuffalo/envy v1.7.0/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI= +github.com/gobuffalo/flect v0.1.0/go.mod h1:d2ehjJqGOH/Kjqcoz+F7jHTBbmDb38yXA598Hb50EGs= +github.com/gobuffalo/flect v0.1.1/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI= +github.com/gobuffalo/flect v0.1.3/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI= +github.com/gobuffalo/genny v0.0.0-20190329151137-27723ad26ef9/go.mod h1:rWs4Z12d1Zbf19rlsn0nurr75KqhYp52EAGGxTbBhNk= +github.com/gobuffalo/genny v0.0.0-20190403191548-3ca520ef0d9e/go.mod h1:80lIj3kVJWwOrXWWMRzzdhW3DsrdjILVil/SFKBzF28= +github.com/gobuffalo/genny v0.1.0/go.mod h1:XidbUqzak3lHdS//TPu2OgiFB+51Ur5f7CSnXZ/JDvo= +github.com/gobuffalo/genny v0.1.1/go.mod h1:5TExbEyY48pfunL4QSXxlDOmdsD44RRq4mVZ0Ex28Xk= +github.com/gobuffalo/gitgen v0.0.0-20190315122116-cc086187d211/go.mod h1:vEHJk/E9DmhejeLeNt7UVvlSGv3ziL+djtTr3yyzcOw= +github.com/gobuffalo/gogen v0.0.0-20190315121717-8f38393713f5/go.mod h1:V9QVDIxsgKNZs6L2IYiGR8datgMhB577vzTDqypH360= +github.com/gobuffalo/gogen v0.1.0/go.mod h1:8NTelM5qd8RZ15VjQTFkAW6qOMx5wBbW4dSCS3BY8gg= +github.com/gobuffalo/gogen v0.1.1/go.mod h1:y8iBtmHmGc4qa3urIyo1shvOD8JftTtfcKi+71xfDNE= +github.com/gobuffalo/logger v0.0.0-20190315122211-86e12af44bc2/go.mod h1:QdxcLw541hSGtBnhUc4gaNIXRjiDppFGaDqzbrBd3v8= +github.com/gobuffalo/mapi v1.0.1/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc= +github.com/gobuffalo/mapi v1.0.2/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc= +github.com/gobuffalo/packd v0.0.0-20190315124812-a385830c7fc0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4= +github.com/gobuffalo/packd v0.1.0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4= +github.com/gobuffalo/packr/v2 v2.0.9/go.mod h1:emmyGweYTm6Kdper+iywB6YK5YzuKchGtJQZ0Odn4pQ= +github.com/gobuffalo/packr/v2 v2.2.0/go.mod h1:CaAwI0GPIAv+5wKLtv8Afwl+Cm78K/I/VCm/3ptBN+0= +github.com/gobuffalo/syncx v0.0.0-20190224160051-33c29581e754/go.mod h1:HhnNqWY95UYwwW3uSASeV7vtgYkT2t16hJgV3AEPUpw= +github.com/gofrs/uuid v3.3.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM= +github.com/gofrs/uuid v4.0.0+incompatible/go.mod h1:b2aQJv3Z4Fp6yNu3cdSllBxTCLRxnplIgP/c0N/04lM= +github.com/gogo/googleapis v1.1.0/go.mod h1:gf4bu3Q80BeJ6H1S1vYPm8/ELATdvryBaNFGgqEef3s= +github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.2.0/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4= +github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o= +github.com/gogo/protobuf v1.2.2-0.20190730201129-28a6bbf47e48/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o= +github.com/gogo/protobuf v1.3.1/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o= +github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= +github.com/golang-sql/civil v0.0.0-20190719163853-cb61b32ac6fe/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0= +github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGwJL78qG/PmXZO1EjYhfJinVAhrmmHX6Z8B9k= +github.com/golang/geo v0.0.0-20190916061304-5b978397cfec/go.mod h1:QZ0nwyI2jOfgRAoBvP+ab5aRr7c9x7lhGEJrKvBwjWI= +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/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/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/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= +github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8= +github.com/golang/protobuf v0.0.0-20161109072736-4bd1920723d7/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.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk= +github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= +github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= +github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= +github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= +github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= +github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= +github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= +github.com/golang/protobuf v1.5.1/go.mod h1:DopwsBzvsk0Fs44TXzsVbJyPhcCPeIwnvohx4u74HPM= +github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= +github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= +github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= +github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +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/flatbuffers v1.11.0/go.mod h1:1AeVuKshWv4vARoZatz6mlQ0JxURH0Kv5+zNeJKJCa8= +github.com/google/gnostic-models v0.6.8 h1:yo/ABAfM5IMRsS1VnXjTBvUb61tFIHozhlYvRgGre9I= +github.com/google/gnostic-models v0.6.8/go.mod h1:5n7qKqH0f5wFt+aWF8CW6pZLLNOfYuF5OpfBSENuI8U= +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.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= +github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= +github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8= +github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/gofuzz v1.1.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0= +github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= +github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= +github.com/google/martian/v3 v3.2.1/go.mod h1:oBOf6HBosgwRXnUGWUB05QECsc6uvmMiJ3+6W4l/CUk= +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-20200417002340-c6e0a841f49a/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20201218002935-b9804c9f04c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210122040257-d980be63207e/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20240525223248-4bfdf5a9a2af h1:kmjWCqn2qkEml422C2Rrd27c3VGxi6a/6HNq8QmHRKM= +github.com/google/pprof v0.0.0-20240525223248-4bfdf5a9a2af/go.mod h1:K1liHPHnj73Fdn/EKuT8nrFqBihUSKXoLYU0BuatOYo= +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.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.1.2/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/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY= +github.com/googleapis/gnostic v0.1.0/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY= +github.com/googleapis/gnostic v0.4.0/go.mod h1:on+2t9HRStVgn95RSsFWFz+6Q0Snyqv1awfrALZdbtU= +github.com/googleapis/gnostic v0.4.1/go.mod h1:LRhVm6pbyptWbWbuZ38d1eyptfvIytN3ir6b65WBswg= +github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g= +github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8= +github.com/gophercloud/gophercloud v0.10.0/go.mod h1:gmC5oQqMDOMO1t1gq5DquX/yAU808e/4mzjjDA76+Ss= +github.com/gophercloud/gophercloud v0.18.0 h1:V6hcuMPmjXg+js9flU8T3RIHDCjV7F5CG5GD0MRhP/w= +github.com/gophercloud/gophercloud v0.18.0/go.mod h1:wRtmUelyIIv3CSSDI47aUwbs075O6i+LY+pXsKCBsb4= +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/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= +github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= +github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc= +github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= +github.com/grafana/grafana-com-public-clients/go/gcom v0.0.0-20241023082751-1bde2051eadd h1:nGQF9y7YA7Ll08s+VujNcmJbstObJ4Q14SqeEkuXm6w= +github.com/grafana/grafana-com-public-clients/go/gcom v0.0.0-20241023082751-1bde2051eadd/go.mod h1:u9d0BESoKlztYm93CpoRleQjMbYBcZ+JOLHHP2nN6Wg= +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/grpc-ecosystem/grpc-gateway v1.14.4/go.mod h1:6CwZWGDSPRJidgKAtJVvND6soZe6fT7iteq8wDPdhb0= +github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= +github.com/hashicorp/consul/api v1.3.0/go.mod h1:MmDNSzIMUjNpY/mQ398R4bk2FnqQLoPndWW5VkKPlCE= +github.com/hashicorp/consul/api v1.4.0/go.mod h1:xc8u05kyMa3Wjr9eEAsIAo3dg8+LywT5E/Cl7cNS5nU= +github.com/hashicorp/consul/api v1.8.1 h1:BOEQaMWoGMhmQ29fC26bi0qb7/rId9JzZP2V0Xmx7m8= +github.com/hashicorp/consul/api v1.8.1/go.mod h1:sDjTOq0yUyv5G4h+BqSea7Fn6BU+XbolEz1952UB+mk= +github.com/hashicorp/consul/sdk v0.3.0/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= +github.com/hashicorp/consul/sdk v0.4.0/go.mod h1:fY08Y9z5SvJqevyZNy6WWPXiG3KwBPAvlcdx16zZ0fM= +github.com/hashicorp/consul/sdk v0.7.0/go.mod h1:fY08Y9z5SvJqevyZNy6WWPXiG3KwBPAvlcdx16zZ0fM= +github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= +github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-cleanhttp v0.5.1 h1:dH3aiDG9Jvb5r5+bYHsikaOUIpcM0xvgMXVoDkXMzJM= +github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-hclog v0.12.2 h1:F1fdYblUEsxKiailtkhCCG2g4bipEgaHiDc8vffNpD4= +github.com/hashicorp/go-hclog v0.12.2/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-immutable-radix v1.2.0 h1:l6UW37iCXwZkZoAbEYnptSHVE/cQ5bOTPYG5W3vf9+8= +github.com/hashicorp/go-immutable-radix v1.2.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= +github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= +github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= +github.com/hashicorp/go-retryablehttp v0.5.3/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs= +github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU= +github.com/hashicorp/go-rootcerts v1.0.2 h1:jzhAVGtqPKbwpyCPELlgNWhE1znq+qwJtW5Oi2viEzc= +github.com/hashicorp/go-rootcerts v1.0.2/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= +github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= +github.com/hashicorp/go-sockaddr v1.0.2/go.mod h1:rB4wwRAUzs07qva3c5SdrY/NEtAUjGlgmH/UkBUC97A= +github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= +github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-version v1.2.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= +github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90= +github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/golang-lru v0.5.4 h1:YDjusn29QI/Das2iO9M0BHnIbxPeyuCHsjMW+lJfyTc= +github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= +github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= +github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= +github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ= +github.com/hashicorp/mdns v1.0.1/go.mod h1:4gW7WsVCke5TE7EPeYliwHlRUyBtfCwuFwuMg2DmyNY= +github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= +github.com/hashicorp/memberlist v0.1.4/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= +github.com/hashicorp/memberlist v0.2.0/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/memberlist v0.2.2/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/memberlist v0.2.3/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= +github.com/hashicorp/serf v0.9.0/go.mod h1:YL0HO+FifKOW2u1ke99DGVu1zhcpZzNwrLIqBC7vbYU= +github.com/hashicorp/serf v0.9.5 h1:EBWvyu9tcRszt3Bxp3KNssBMP1KuHWyO51lz9+786iM= +github.com/hashicorp/serf v0.9.5/go.mod h1:UWDWwZeL5cuWDJdl0C6wrvrUwEqtQ4ZKBKKENpqIUyk= +github.com/hetznercloud/hcloud-go v1.26.2 h1:fI8BXAGJI4EFeCDd2a/I4EhqyK32cDdxGeWfYMGUi50= +github.com/hetznercloud/hcloud-go v1.26.2/go.mod h1:2C5uMtBiMoFr3m7lBFPf7wXTdh33CevmZpQIIDPGYJI= +github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/huandu/xstrings v1.0.0/go.mod h1:4qWG/gcEcfX4z/mBDHJ++3ReCw9ibxbsNJbcucJdbSo= +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/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= +github.com/imdario/mergo v0.3.4/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA= +github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA= +github.com/imdario/mergo v0.3.15 h1:M8XP7IuFNsqUx6VPK2P9OSmsYsI/YFaGil0uD21V3dM= +github.com/imdario/mergo v0.3.15/go.mod h1:WBLT9ZmE3lPoWsEzCh9LPo3TiwVN+ZKEjmz+hD27ysY= +github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= +github.com/influxdata/flux v0.65.0/go.mod h1:BwN2XG2lMszOoquQaFdPET8FRQfrXiZsWmcMO9rkaVY= +github.com/influxdata/flux v0.113.0/go.mod h1:3TJtvbm/Kwuo5/PEo5P6HUzwVg4bXWkb2wPQHPtQdlU= +github.com/influxdata/httprouter v1.3.1-0.20191122104820-ee83e2772f69/go.mod h1:pwymjR6SrP3gD3pRj9RJwdl1j5s3doEEV8gS4X9qSzA= +github.com/influxdata/influxdb v1.8.0/go.mod h1:SIzcnsjaHRFpmlxpJ4S3NT64qtEKYweNTUMb/vh0OMQ= +github.com/influxdata/influxdb v1.9.2/go.mod h1:UEe3MeD9AaP5rlPIes102IhYua3FhIWZuOXNHxDjSrI= +github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= +github.com/influxdata/influxql v1.1.0/go.mod h1:KpVI7okXjK6PRi3Z5B+mtKZli+R1DnZgb3N+tzevNgo= +github.com/influxdata/influxql v1.1.1-0.20210223160523-b6ab99450c93/go.mod h1:gHp9y86a/pxhjJ+zMjNXiQAA197Xk9wLxaz+fGG+kWk= +github.com/influxdata/line-protocol v0.0.0-20180522152040-32c6aa80de5e/go.mod h1:4kt73NQhadE3daL3WhR5EJ/J2ocX0PZzwxQ0gXJ7oFE= +github.com/influxdata/pkg-config v0.2.6/go.mod h1:EMS7Ll0S4qkzDk53XS3Z72/egBsPInt+BeRxb0WeSwk= +github.com/influxdata/pkg-config v0.2.7/go.mod h1:EMS7Ll0S4qkzDk53XS3Z72/egBsPInt+BeRxb0WeSwk= +github.com/influxdata/promql/v2 v2.12.0/go.mod h1:fxOPu+DY0bqCTCECchSRtWfc+0X19ybifQhZoQNF5D8= +github.com/influxdata/roaring v0.4.13-0.20180809181101-fc520f41fab6/go.mod h1:bSgUQ7q5ZLSO+bKBGqJiCBGAl+9DxyW63zLTujjUlOE= +github.com/influxdata/tdigest v0.0.0-20181121200506-bf2b5ad3c0a9/go.mod h1:Js0mqiSBE6Ffsg94weZZ2c+v/ciT8QRHFOap7EKDrR0= +github.com/influxdata/tdigest v0.0.2-0.20210216194612-fc98d27c9e8b/go.mod h1:Z0kXnxzbTC2qrx4NaIzYkE1k66+6oEDQTvL95hQFh5Y= +github.com/influxdata/usage-client v0.0.0-20160829180054-6d3895376368/go.mod h1:Wbbw6tYNvwa5dlB6304Sd+82Z3f7PmVZHVKU637d4po= +github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= +github.com/jessevdk/go-flags v1.5.0/go.mod h1:Fw0T6WPc1dYxT4mKEZRfG5kJhaTDP9pj1c2EWnYs/m4= +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/jmespath/go-jmespath v0.4.0 h1:BEgLn5cpjn8UN1mAw4NjwDrS35OdebyEtFe+9YPoQUg= +github.com/jmespath/go-jmespath v0.4.0/go.mod h1:T8mJZnbsbmF+m6zOOFylbeCJqk5+pHWvzYPziyZiYoo= +github.com/jmespath/go-jmespath/internal/testify v1.5.1/go.mod h1:L3OGu8Wl2/fWfCI6z80xFu9LTZmf1ZRjMHUOPmWr69U= +github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg= +github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= +github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY= +github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= +github.com/jpillora/backoff v1.0.0 h1:uvFg412JmmHBHw7iwprIxkPMI+sGQ4kzOWsMeHnm2EA= +github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= +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/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +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= +github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= +github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= +github.com/jsternberg/zap-logfmt v1.0.0/go.mod h1:uvPs/4X51zdkcm5jXl5SYoN+4RK21K8mysFmDaM/h+o= +github.com/jsternberg/zap-logfmt v1.2.0/go.mod h1:kz+1CUmCutPWABnNkOu9hOHKdT2q3TDYCcsFy9hpqb0= +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/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= +github.com/jung-kurt/gofpdf v1.0.3-0.20190309125859-24315acbbda5/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= +github.com/jwilder/encoding v0.0.0-20170811194829-b4e1701a28ef/go.mod h1:Ct9fl0F6iIOGgxJ5npU/IUOhOhqlVrGjyIZc8/MagT0= +github.com/karrick/godirwalk v1.8.0/go.mod h1:H5KPZjojv4lE+QYImBI8xVtrBRgYrIVsaRPx4tDPEn4= +github.com/karrick/godirwalk v1.10.3/go.mod h1:RoGL9dQei4vP9ilrpETWE8CLOZ1kiN0LhBygSwrAsHA= +github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= +github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= +github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/klauspost/compress v1.4.0/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A= +github.com/klauspost/compress v1.9.5/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A= +github.com/klauspost/cpuid v0.0.0-20170728055534-ae7887de9fa5/go.mod h1:Pj4uuM528wm8OyEC2QMXAi2YiTZ96dNQPGgoMS4s3ek= +github.com/klauspost/crc32 v0.0.0-20161016154125-cb6bfca970f6/go.mod h1:+ZoRqAPRLkC4NPOvfYeR5KNOrY6TD+/sAC3HXPZgDYg= +github.com/klauspost/pgzip v1.0.2-0.20170402124221-0bf5dcad4ada/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs= +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/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= +github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= +github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= +github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= +github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/pty v1.1.5/go.mod h1:9r2w37qlBe7rQ6e1fg1S/9xpWHSnaqNdHD3WcMdbPDA= +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/kylelemons/godebug v0.0.0-20160406211939-eadb3ce320cb/go.mod h1:B69LEHPfb2qLo0BaaOLcbitczOKLWTsrBG9LczfCD4k= +github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= +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/linode/linodego v0.28.5 h1:JaCziTxHJ7a01MYyjHqskRc8zXQxXOddwrDeoQ2rBnw= +github.com/linode/linodego v0.28.5/go.mod h1:BR0gVkCJffEdIGJSl6bHR80Ty+Uvg/2jkjmrWaFectM= +github.com/lyft/protoc-gen-validate v0.0.13/go.mod h1:XbGvPuh87YZc5TdIa2/I4pLk0QoUACkjt2znoq26NVQ= +github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= +github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/mailru/easyjson v0.0.0-20180823135443-60711f1a8329/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +github.com/mailru/easyjson v0.0.0-20190312143242-1de009706dbe/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= +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/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs= +github.com/mailru/easyjson v0.7.1/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs= +github.com/mailru/easyjson v0.7.6/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= +github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= +github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= +github.com/markbates/oncer v0.0.0-20181203154359-bf2de49a0be2/go.mod h1:Ld9puTsIW75CHf65OeIOkyKbteujpZVXDpWK6YGZbxE= +github.com/markbates/safe v1.0.1/go.mod h1:nAqgmRi7cY2nqMc92/bSEeQA+R4OheNU2T1kNSCBdG0= +github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= +github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= +github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= +github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= +github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= +github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= +github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= +github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= +github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= +github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= +github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= +github.com/mattn/go-isatty v0.0.19/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.3/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= +github.com/mattn/go-sqlite3 v1.11.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc= +github.com/mattn/go-tty v0.0.0-20180907095812-13ff1204f104/go.mod h1:XPvLUNfbS4fJH25nqRHfWLMa1ONC8Amw+mIA639KxkE= +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/miekg/dns v1.1.22/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= +github.com/miekg/dns v1.1.26/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= +github.com/miekg/dns v1.1.29/go.mod h1:KNUDUusw/aVsxyTYZM1oqvCicbwhgbNgztCETuNZ7xM= +github.com/miekg/dns v1.1.42/go.mod h1:+evo5L0630/F6ca/Z9+GAqzhjGyn8/c+TBaOyfEl0V4= +github.com/miekg/dns v1.1.56 h1:5imZaSeoRNvpM9SzWNhEcP9QliKiz20/dA2QabIGVnE= +github.com/miekg/dns v1.1.56/go.mod h1:cRm6Oo2C8TY9ZS/TqsSrseAcncm74lfK5G+ikN2SWWY= +github.com/mileusna/useragent v0.0.0-20190129205925-3e331f0949a5/go.mod h1:JWhYAp2EXqUtsxTKdeGlY8Wp44M7VxThC9FEoNGi2IE= +github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= +github.com/mitchellh/cli v1.1.0/go.mod h1:xcISNoH86gajksDmfB23e/pu+B+GeFRMYmoHXxx3xhI= +github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= +github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= +github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= +github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= +github.com/mitchellh/go-wordwrap v1.0.0/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo= +github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg= +github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY= +github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/mitchellh/mapstructure v1.2.2/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/mitchellh/mapstructure v1.3.2/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/mitchellh/mapstructure v1.3.3/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/mitchellh/mapstructure v1.4.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/mitchellh/mapstructure v1.4.1 h1:CpVNEelQCZBooIPDn+AR3NpivK/TIKU8bDxdASFVQag= +github.com/mitchellh/mapstructure v1.4.1/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/moby/spdystream v0.2.0/go.mod h1:f7i0iNDQJ059oMTcWxx8MA/zKFIuD/lY+0GqbN2Wy8c= +github.com/moby/spdystream v0.4.0 h1:Vy79D6mHeJJjiPdFEL2yku1kl0chZpJfZcPpb16BRl8= +github.com/moby/spdystream v0.4.0/go.mod h1:xBAYlnt/ay+11ShkdFKNAG7LsyK/tmNBVvVOwrfMgdI= +github.com/moby/term v0.0.0-20201216013528-df9cb8a40635/go.mod h1:FBS0z0QWA44HXygs7VXDUOGoN/1TV3RuWkLO04am3wc= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= +github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= +github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc= +github.com/morikuni/aec v1.0.0/go.mod h1:BbKIizmSmc5MMPqRYbxO4ZU0S0+P200+tUnFx7PXmsc= +github.com/mschoch/smat v0.0.0-20160514031455-90eadee771ae/go.mod h1:qAyveg+e4CE+eKJXWVjKXM4ck2QobLqTDytGJbLLhJg= +github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= +github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA= +github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= +github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f h1:KUppIJq7/+SVif2QVs3tOP0zanoHgBEVAwHxUSIzRqU= +github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f h1:y5//uYreIhSUg3J1GEMiLbxo1LJaP8RfCpH6pymGZus= +github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw= +github.com/nats-io/jwt v0.3.0/go.mod h1:fRYCDE99xlTsqUzISS1Bi75UBJ6ljOJQOAAu5VglpSg= +github.com/nats-io/jwt v0.3.2/go.mod h1:/euKqTS1ZD+zzjYrY7pseZrTtWQSjujC7xjPc8wL6eU= +github.com/nats-io/nats-server/v2 v2.1.2/go.mod h1:Afk+wRZqkMQs/p45uXdrVLuab3gwv3Z8C4HTBu8GD/k= +github.com/nats-io/nats.go v1.9.1/go.mod h1:ZjDU1L/7fJ09jvUSRVBR2e7+RnLiiIQyqyzEE/Zbp4w= +github.com/nats-io/nkeys v0.1.0/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= +github.com/nats-io/nkeys v0.1.3/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= +github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= +github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= +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/oklog/run v1.1.0/go.mod h1:sVPdnTZT1zYwAJeCMu2Th4T21pA3FPOQRfWjQlk7DVU= +github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U= +github.com/olekukonko/tablewriter v0.0.0-20170122224234-a0225b3f23b5/go.mod h1:vsDQFd/mU46D+Z4whnwzcISnGGzXWMclvtLoiIKAKIo= +github.com/onsi/ginkgo v0.0.0-20170829012221-11459a886d9c/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +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/ginkgo v1.11.0 h1:JAKSXpt1YjtLA7YpPiqO9ss6sNXEsPfSGdwN0UHqzrw= +github.com/onsi/ginkgo v1.11.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo/v2 v2.19.0 h1:9Cnnf7UHo57Hy3k6/m5k3dRfGTMXGvxhHFvkDTCTpvA= +github.com/onsi/ginkgo/v2 v2.19.0/go.mod h1:rlwLi9PilAFJ8jCg9UE1QP6VBpd6/xj3SRC0d6TU0To= +github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA= +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/onsi/gomega v1.33.1 h1:dsYjIxxSR755MDmKVsaFQTE22ChNBcuuTWgkUDSubOk= +github.com/onsi/gomega v1.33.1/go.mod h1:U4R44UsT+9eLIaYRB2a5qajjtQYn0hauxvRm16AVYg0= +github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk= +github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U= +github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM= +github.com/opencontainers/image-spec v1.0.1 h1:JMemWkRwHx4Zj+fVxWoMCFm/8sYGGrUVojFA6h/TRcI= +github.com/opencontainers/image-spec v1.0.1/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= +github.com/opentracing-contrib/go-observer v0.0.0-20170622124052-a52f23424492/go.mod h1:Ngi6UdF0k5OKD5t5wlmGhe/EDKPoUM3BXZSSfIuJbis= +github.com/opentracing-contrib/go-stdlib v0.0.0-20190519235532-cf7a6c988dc9/go.mod h1:PLldrQSroqzH70Xl+1DQcGnefIbqsKR7UDaiux3zV+w= +github.com/opentracing-contrib/go-stdlib v1.0.0/go.mod h1:qtI1ogk+2JhVPIXVc6q+NHziSmy2W5GbdQZFUHADCBU= +github.com/opentracing/basictracer-go v1.0.0/go.mod h1:QfBfYuafItcjQuMwinw9GhYKwFXS9KnPs5lxoYwgW74= +github.com/opentracing/opentracing-go v1.0.2/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= +github.com/opentracing/opentracing-go v1.0.3-0.20180606204148-bd9c31933947/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= +github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= +github.com/opentracing/opentracing-go v1.2.0/go.mod h1:GxEUsuufX4nBwe+T+Wl9TAgYrxe9dPLANfrWvHYVTgc= +github.com/openzipkin-contrib/zipkin-go-opentracing v0.4.5/go.mod h1:/wsWhb9smxSfWAKL3wpBW7V8scJMt8N8gnaMCS9E/cA= +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/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/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= +github.com/paulbellamy/ratecounter v0.2.0/go.mod h1:Hfx1hDpSGoqxkVVpBi/IlYD7kChlfo5C6hzIHwPqfFE= +github.com/pborman/uuid v1.2.0/go.mod h1:X/NO0urCmaxf9VXbdlT7C2Yzkj2IKimNn4k+gtPdI/k= +github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= +github.com/pelletier/go-toml v1.4.0/go.mod h1:PN7xzY2wHTK0K9p34ErDQMlFxa51Fk0OUruD3k1mMwo= +github.com/pelletier/go-toml v1.7.0/go.mod h1:vwGMzjaWMwyfHwgIBhI2YUM4fB6nL6lVAvS1LBMMhTE= +github.com/performancecopilot/speed v3.0.0+incompatible/go.mod h1:/CLtqpZ5gBg1M9iaPbIdPPGyKcA8hKdoy6hAWba7Yac= +github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU= +github.com/peterh/liner v1.0.1-0.20180619022028-8c1271fcf47f/go.mod h1:xIteQHvHuaLYG9IFj6mSxM0fCKrs34IrEQUhOYuGPHc= +github.com/philhofer/fwd v1.0.0/go.mod h1:gk3iGcWd9+svBvR0sR+KPcfE+RNWozjowpeBVG3ZVNU= +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/browser v0.0.0-20180916011732-0a3d74bf9ce4/go.mod h1:4OwLy04Bl9Ef3GJJCoec+30X3LQs/0/m4HFRt/2LUSA= +github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= +github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/profile v1.2.1/go.mod h1:hJw3o1OdXxsrSjjVksARp5W95eeEaEfptyVZyv6JUPA= +github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg= +github.com/pkg/term v0.0.0-20180730021639-bffc007b7fd5/go.mod h1:eCbImbZ95eXtAUIbLAuAVnBnwf83mjf6QIVH8SHYwqQ= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U= +github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= +github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s= +github.com/pquerna/cachecontrol v0.0.0-20171018203845-0dec1b30a021/go.mod h1:prYjPmNq4d1NPVmpShWobRqXY3q7Vp+80DqgxxUrUIA= +github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring v0.50.0 h1:eIYVhtUPLDah0nhcHaWItFM595UAGVFKECaWoW02FUA= +github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring v0.50.0/go.mod h1:3WYi4xqXxGGXWDdQIITnLNmuDzO5n6wYva9spVhR4fg= +github.com/prometheus/alertmanager v0.20.0/go.mod h1:9g2i48FAyZW6BtbsnvHtMHQXl2aVtrORKwKVCQ+nbrg= +github.com/prometheus/alertmanager v0.22.2/go.mod h1:rYinOWxFuCnNssc3iOjn2oMTlhLaPcUuqV5yk5JKUAE= +github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= +github.com/prometheus/client_golang v0.9.3-0.20190127221311-3c4408c8b829/go.mod h1:p2iRAGwDERtqlqzRXnrOVns+ignqQo//hLXqYxZYVNs= +github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= +github.com/prometheus/client_golang v1.2.1/go.mod h1:XMU6Z2MjaRKVu/dC1qupJI9SiNkDYzz3xecMgSW/F+U= +github.com/prometheus/client_golang v1.3.0/go.mod h1:hJaj2vgQTGQmVCsAACORcieXFeDPbaTKGT+JTgUa3og= +github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= +github.com/prometheus/client_golang v1.5.1/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= +github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= +github.com/prometheus/client_golang v1.10.0/go.mod h1:WJM3cc3yu7XKBKa/I8WeZm+V3eltZnBwfENSU7mdogU= +github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= +github.com/prometheus/client_golang v1.19.1 h1:wZWJDwK+NameRJuPGDhlnFgx8e8HN3XHQeLaYJFJBOE= +github.com/prometheus/client_golang v1.19.1/go.mod h1:mP78NwGzrVks5S2H6ab8+ZZGJLZUq1hoULYBAYBw1Ho= +github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190115171406-56726106282f/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.1.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.6.1 h1:ZKSh/rekM+n3CeS952MLRAdFwIKqeY8b62p8ais2e9E= +github.com/prometheus/client_model v0.6.1/go.mod h1:OrxVMOVHjw3lKMa8+x6HeMGkHMQyHDk9E3jmP2AmGiY= +github.com/prometheus/common v0.2.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.6.0/go.mod h1:eBmuwkDJBwy6iBfxCBob6t6dR6ENT/y+J+Zk0j9GMYc= +github.com/prometheus/common v0.7.0/go.mod h1:DjGbpBbp5NYNiECxcL/VnbXCCaQpKd3tt26CguLLsqA= +github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= +github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= +github.com/prometheus/common v0.15.0/go.mod h1:U+gB1OBLb1lF3O42bTCL+FK18tX9Oar16Clt/msog/s= +github.com/prometheus/common v0.18.0/go.mod h1:U+gB1OBLb1lF3O42bTCL+FK18tX9Oar16Clt/msog/s= +github.com/prometheus/common v0.23.0/go.mod h1:H6QK/N6XVT42whUeIdI3dp36w49c+/iMDk7UAI2qm7Q= +github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= +github.com/prometheus/common v0.29.0/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= +github.com/prometheus/common v0.55.0 h1:KEi6DK7lXW/m7Ig5i47x0vRzuBsHuvJdi5ee6Y3G1dc= +github.com/prometheus/common v0.55.0/go.mod h1:2SECS4xJG1kd8XF9IcM1gMX6510RAEL65zxzNImwdc8= +github.com/prometheus/exporter-toolkit v0.5.1/go.mod h1:OCkM4805mmisBhLmVFw858QYi3v0wKdY6/UxrT0pZVg= +github.com/prometheus/exporter-toolkit v0.6.0/go.mod h1:ZUBIj498ePooX9t/2xtDjeQYwvRpiPP2lh5u4iblj2g= +github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/prometheus/procfs v0.0.5/go.mod h1:4A/X28fw3Fc593LaREMrKMqOKvUAntwMDaekg4FpcdQ= +github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= +github.com/prometheus/procfs v0.0.11/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= +github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= +github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= +github.com/prometheus/procfs v0.15.1 h1:YagwOFzUgYfKKHX6Dr+sHT7km/hxC76UB0learggepc= +github.com/prometheus/procfs v0.15.1/go.mod h1:fB45yRUv8NstnjriLhBQLuOUt+WW4BsoGhij/e3PBqk= +github.com/prometheus/prometheus v0.0.0-20200609090129-a6600f564e3c/go.mod h1:S5n0C6tSgdnwWshBUceRx5G1OsjLv/EeZ9t3wIfEtsY= +github.com/prometheus/prometheus v1.8.2-0.20210701133801-b0944590a1c9 h1:If7jYp33vwa8ZQ7GGwrAs0SBjiW0aWeAB/oV1aG7bZ4= +github.com/prometheus/prometheus v1.8.2-0.20210701133801-b0944590a1c9/go.mod h1:A97P+iwS3Ffpxpejz4+ASZl6i9EqSJDzxObq8DjV2SU= +github.com/rcrowley/go-metrics v0.0.0-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= +github.com/retailnext/hllpp v1.0.1-0.20180308014038-101a6d2f8b52/go.mod h1:RDpi1RftBQPUCDRw6SmxeaREsAaRKnOclghuzp/WRzc= +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.1.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.2.2/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/rogpeppe/go-internal v1.12.0 h1:exVL4IDcn6na9z1rAb56Vxr+CgyK3nn3O+epU5NdKM8= +github.com/rogpeppe/go-internal v1.12.0/go.mod h1:E+RYuTGaKKdloAfM02xzb0FW3Paa99yedzYV+kq4uf4= +github.com/rs/cors v1.6.0/go.mod h1:gFx+x8UowdsKA9AchylcLynDq+nNFfI8FkUZdN/jGCU= +github.com/rs/cors v1.7.0/go.mod h1:gFx+x8UowdsKA9AchylcLynDq+nNFfI8FkUZdN/jGCU= +github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= +github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= +github.com/ryanuber/columnize v2.1.0+incompatible/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= +github.com/samuel/go-zookeeper v0.0.0-20190923202752-2cc03de413da/go.mod h1:gi+0XIa01GRL2eRQVjQkKGqKF3SF9vZR/HnPullcV2E= +github.com/satori/go.uuid v0.0.0-20160603004225-b111a074d5ef/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= +github.com/satori/go.uuid v1.2.1-0.20181028125025-b2ce2384e17b/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= +github.com/scaleway/scaleway-sdk-go v1.0.0-beta.7.0.20210223165440-c65ae3540d44 h1:3egqo0Vut6daANFm7tOXdNAa8v5/uLU+sgCJrc88Meo= +github.com/scaleway/scaleway-sdk-go v1.0.0-beta.7.0.20210223165440-c65ae3540d44/go.mod h1:CJJ5VAbozOl0yEw7nHB9+7BXTJbIn6h7W+f6Gau5IP8= +github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= +github.com/segmentio/kafka-go v0.1.0/go.mod h1:X6itGqS9L4jDletMsxZ7Dz+JFWxM6JHfPOCvTvk+EJo= +github.com/segmentio/kafka-go v0.2.0/go.mod h1:X6itGqS9L4jDletMsxZ7Dz+JFWxM6JHfPOCvTvk+EJo= +github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= +github.com/shurcooL/httpfs v0.0.0-20190707220628-8d4bc4ba7749/go.mod h1:ZY1cvUeJuFPAdZ/B6v7RHavJWZn2YPVFQ1OSXhCGOkg= +github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +github.com/shurcooL/vfsgen v0.0.0-20181202132449-6a9ea43bcacd/go.mod h1:TrYk7fJVaAttu97ZZKrO9UbRa8izdowaMIZcxYMbVaw= +github.com/shurcooL/vfsgen v0.0.0-20200824052919-0d455de96546/go.mod h1:TrYk7fJVaAttu97ZZKrO9UbRa8izdowaMIZcxYMbVaw= +github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= +github.com/sirupsen/logrus v1.4.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.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= +github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= +github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +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/snowflakedb/gosnowflake v1.3.4/go.mod h1:NsRq2QeiMUuoNUJhp5Q6xGC4uBrsS9g6LwZVEkTWgsE= +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/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= +github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= +github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk= +github.com/spf13/afero v1.8.2 h1:xehSyVa0YnHWsJ49JFljMpg1HX19V6NDZ1fkm1Xznbo= +github.com/spf13/afero v1.8.2/go.mod h1:CtAatgMJh6bJEIs48Ay/FOnkljP3WeGUG0MC1RfAqwo= +github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= +github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= +github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU= +github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo= +github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.1/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s= +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= +github.com/streadway/handy v0.0.0-20190108123426-d5acb3125c2a/go.mod h1:qNTQ5P5JnDBl6z3cMAg/SywNDC5ABu5ApDIw6lUbRmI= +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/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/testify v1.2.0/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= +github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= +github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +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/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk= +github.com/tinylib/msgp v1.0.2/go.mod h1:+d+yLhGm8mzTaHzB+wgMYrodPfmZrzkirds8fDWklFE= +github.com/tinylib/msgp v1.1.0/go.mod h1:+d+yLhGm8mzTaHzB+wgMYrodPfmZrzkirds8fDWklFE= +github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= +github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= +github.com/uber-go/tally v3.3.15+incompatible/go.mod h1:YDTIBxdXyOU/sCWilKB4bgyufu1cEi0jdVnRdxvjnmU= +github.com/uber/athenadriver v1.1.4/go.mod h1:tQjho4NzXw55LGfSZEcETuYydpY1vtmixUabHkC1K/E= +github.com/uber/jaeger-client-go v2.23.0+incompatible/go.mod h1:WVhlPFC8FDjOFMMWRy2pZqQJSXxYSwNYOkTr/Z6d3Kk= +github.com/uber/jaeger-client-go v2.29.1+incompatible/go.mod h1:WVhlPFC8FDjOFMMWRy2pZqQJSXxYSwNYOkTr/Z6d3Kk= +github.com/uber/jaeger-lib v2.2.0+incompatible/go.mod h1:ComeNDZlWwrWnDv8aPp0Ba6+uUTzImX/AauajbLI56U= +github.com/uber/jaeger-lib v2.4.1+incompatible/go.mod h1:ComeNDZlWwrWnDv8aPp0Ba6+uUTzImX/AauajbLI56U= +github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= +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/vektah/gqlparser v1.1.2/go.mod h1:1ycwN7Ij5njmMkPPAOaRFY4rET2Enx7IkVv3vaXspKw= +github.com/vladimirvivien/gexe v0.3.0 h1:4xwiOwGrDob5OMR6E92B9olDXYDglXdHhzR1ggYtWJM= +github.com/vladimirvivien/gexe v0.3.0/go.mod h1:fp7cy60ON1xjhtEI/+bfSEIXX35qgmI+iRYlGOqbBFM= +github.com/willf/bitset v1.1.3/go.mod h1:RjeCKbqT1RxIR/KWY6phxZiaY1IyutSBfGjNPySAYV4= +github.com/willf/bitset v1.1.9/go.mod h1:RjeCKbqT1RxIR/KWY6phxZiaY1IyutSBfGjNPySAYV4= +github.com/x448/float16 v0.8.4 h1:qLwI1I70+NjRFUR3zs1JPUCgaCXSh3SW62uAKT1mSBM= +github.com/x448/float16 v0.8.4/go.mod h1:14CWIYCyZA/cWjXOioeEpHeN/83MdbZDRQHoFcYsOfg= +github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= +github.com/xdg-go/scram v1.0.2/go.mod h1:1WAq6h33pAW+iRreB34OORO2Nf7qel3VV3fjBj+hCSs= +github.com/xdg-go/stringprep v1.0.2/go.mod h1:8F9zXuvzgwmyT5DUm4GUfZGDdT3W+LCvS6+da4O5kxM= +github.com/xdg/scram v0.0.0-20180814205039-7eeb5667e42c/go.mod h1:lB8K/P019DLNhemzwFU4jHLhdvlE6uDZjXFejJXr49I= +github.com/xdg/stringprep v0.0.0-20180714160509-73f8eece6fdc/go.mod h1:Jhud4/sHMO4oL310DaZAKk9ZaJ08SJfe+sJh0HrGL1Y= +github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= +github.com/xlab/treeprint v0.0.0-20180616005107-d6fb6747feb6/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg= +github.com/xlab/treeprint v1.1.0/go.mod h1:gj5Gd3gPdKtR1ikdDK6fnFLdmIS0X30kTTuNd/WEJu0= +github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= +github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA= +github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +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= +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.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM= +go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM= +go.mongodb.org/mongo-driver v1.1.2/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM= +go.mongodb.org/mongo-driver v1.3.0/go.mod h1:MSWZXKOynuguX+JSvwP8i+58jYCXxbia8HS3gZBapIE= +go.mongodb.org/mongo-driver v1.3.2/go.mod h1:MSWZXKOynuguX+JSvwP8i+58jYCXxbia8HS3gZBapIE= +go.mongodb.org/mongo-driver v1.3.4/go.mod h1:MSWZXKOynuguX+JSvwP8i+58jYCXxbia8HS3gZBapIE= +go.mongodb.org/mongo-driver v1.4.3/go.mod h1:WcMNYLx/IlOxLe6JRJiv2uXuCz6zBLndR4SoGjYphSc= +go.mongodb.org/mongo-driver v1.4.4/go.mod h1:WcMNYLx/IlOxLe6JRJiv2uXuCz6zBLndR4SoGjYphSc= +go.mongodb.org/mongo-driver v1.4.6/go.mod h1:WcMNYLx/IlOxLe6JRJiv2uXuCz6zBLndR4SoGjYphSc= +go.mongodb.org/mongo-driver v1.5.1/go.mod h1:gRXCHX4Jo7J0IJ1oDQyUxF7jfy19UfxniMS4xxMmUqw= +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.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.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= +go.uber.org/atomic v1.5.1/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= +go.uber.org/atomic v1.6.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= +go.uber.org/atomic v1.8.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A= +go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= +go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= +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.4.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= +go.uber.org/multierr v1.5.0/go.mod h1:FeouvMocqHpRaaGuG9EjoKcStLC43Zu/fmqdUMPcKYU= +go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= +go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= +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.14.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= +go.uber.org/zap v1.14.1/go.mod h1:Mb2vm2krFEG5DV0W9qcHBYFtp/Wku1cvYaqPsS/WYfc= +go.uber.org/zap v1.26.0 h1:sI7k6L95XOKS281NhVKOFCUNIvv9e0w4BF8N3u+tCRo= +go.uber.org/zap v1.26.0/go.mod h1:dtElttAiwGvoJ/vj4IwHBS/gXsEu/pZ50mUIRWuG0so= +golang.org/x/crypto v0.0.0-20180505025534-4ec37c66abab/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-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20190211182817-74369b46fc67/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-20190320223903-b7391e95e576/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190325154230-a5d413f7728c/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190422162423-af44ce270edf/go.mod h1:WFFai1msRO1wXaEeE5yQxYXgSfI8pQAWXbQop6sCtWE= +golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190530122614-20be4c3c3ed5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190617133340-57b3e21c3d56/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY= +golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20191202143827-86a70503ff7e/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20191206172530-e9b2fee46413/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20200220183623-bac4c82f6975/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20200422194213-44a606286825/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +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-20201208171446-5f87f3452ae9/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= +golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= +golang.org/x/crypto v0.0.0-20210220033148-5ea612d1eb83/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= +golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= +golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20211108221036-ceb1ce70b4fa/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.24.0 h1:mnl8DM0o513X8fdIkmyFE/5hTYxbwYOjDS/+rK6qpRI= +golang.org/x/crypto v0.24.0/go.mod h1:Z1PMYSOR5nyMcyAVAIQSKCDwalqy85Aqn1x3Ws4L5DM= +golang.org/x/exp v0.0.0-20180321215751-8460e604b9de/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20180807140117-3d87b88a115f/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +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-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= +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-20230515195305-f3d0a9c9a5cc h1:mCRnTeVUjcrhlRmO0VK8a6k6Rrf6TF9htwo2pJVSjIU= +golang.org/x/exp v0.0.0-20230515195305-f3d0a9c9a5cc/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w= +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= +golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= +golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= +golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= +golang.org/x/lint v0.0.0-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/lint v0.0.0-20201208152925-83fdc39ff7b5/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= +golang.org/x/lint v0.0.0-20210508222113-6edffad5e616/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= +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.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.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.17.0 h1:zY54UmvipHiNd+pm+m0x9KhZ9hl1/7QNMyxXbc6ICqA= +golang.org/x/mod v0.17.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= +golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181005035420-146acd28ed58/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190125091013-d26f9f9a57f3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190320064053-1272bf9dcd53/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-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-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20191002035440-2ec189313ef0/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20191004110552-13f9640d40b9/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20191126235420-ef20fe5d7933/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-20200421231249-e086a090c8fd/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-20200506145744-7e3656a0809f/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200513185701-a91f0712d120/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200520182314-0ba52f642ac2/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200602114024-627f9648deb9/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210224082022-3d97a244fca7/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= +golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20210421230115-4e50805a0758/go.mod h1:72T/g9IO56b78aLF+1Kcs5dz7/ng1VjMUvfKvpfy+jM= +golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20210610132358-84b48f89b13b/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= +golang.org/x/net v0.26.0 h1:soB7SVo0PWrY4vPW/+ay0jKDNScG2X9wFeYlXIvJsOQ= +golang.org/x/net v0.26.0/go.mod h1:5YKkiSynbBIh3p6iOc/vibscux0x38BZDkn8sCUPxHE= +golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/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/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210220000619-9bb904979d93/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210313182246-cd4f82c27b84/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.21.0 h1:tsimM75w1tF/uws5rbeHzIWxEqElMehnc+iW793zsZs= +golang.org/x/oauth2 v0.21.0/go.mod h1:XYTD2NtWslqkgxebSiOHnXEap4TF09sJSc7H1sXbhtI= +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-20190412183630-56d357773e84/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-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M= +golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= +golang.org/x/sys v0.0.0-20170830134202-bb24a47a89ea/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +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-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181122145206-62eef0e2fa9b/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190209173611-3b5209105503/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-20190321052220-f7bb7a8bee54/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-20190419153524-e8e3143a4f4a/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-20190531175056-4c3a928424d2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190616124812-15dcb6c0061f/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-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191022100944-742c48ecaeb7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191128015809-6d18c012aee9/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-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200107162124-548cf772de50/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-20200124204421-9fbb57f87de9/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-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200420163511-1957bb5e6d1f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200831180312-196b9ba8737a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/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-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210225134936-a50acf3fe073/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210309074719-68d13333faf2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.21.0 h1:rF+pYz3DAGSQAxAu1CbC7catZg4ebC4UIeIhKxBZvws= +golang.org/x/sys v0.21.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +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-20210220032956-6a3ed077a48d/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.21.0 h1:WVXCp+/EBEHOj53Rvu+7KiT/iElMrO8ACK16SMZ3jaA= +golang.org/x/term v0.21.0/go.mod h1:ooXLefLobQVslOqselCNF4SxFAaoS6KujMbsGzSDmX0= +golang.org/x/text v0.0.0-20160726164857-2910a502d2bf/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +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.4/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.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.16.0 h1:a94ExnEXNtEwYLGJSIUxnWoxoRz/ZcCsV63ROupILh4= +golang.org/x/text v0.16.0/go.mod h1:GhwF1Be+LQoKShO3cGOHzqOgRrGaYc9AvblQOmPVHnI= +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/time v0.0.0-20200416051211-89c76fbcd5d1/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20210611083556-38a9dc6acbc6/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.4.0 h1:Z81tqI5ddIoXDPvVQ7/7CC9TnLM7ubaFG2qXYd5BbYY= +golang.org/x/time v0.4.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= +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-20181011042414-1f849cf54d09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/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-20190125232054-d66bd3c5d5a6/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= +golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +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-20190329151228-23e29df326fe/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190416151739-9c9e1878f421/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190420181800-aa740d480789/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-20190531172133-b3315ee88b7d/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= +golang.org/x/tools v0.0.0-20190617190820-da514acc4774/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-20190624222133-a101b041ded4/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-20190813034749-528a2984e271/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/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-20190920225731-5eefd052ad72/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-20191108193012-7d206e10da11/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-20191203134012-c197fd4bf371/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191216052735-49a3e744a425/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-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-20200108203644-89082a384178/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-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200304024140-c4206d458c3f/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= +golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= +golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= +golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= +golang.org/x/tools v0.0.0-20200422205258-72e4a01eba43/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.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200721032237-77f530d86f9a/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE= +golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210108195828-e2f9c7f1fc8e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= +golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.3/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d h1:vU5i/LfpvrRCpgM/VPfJLg5KjxD3E+hfT1SH+d9zLwg= +golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gomodules.xyz/jsonpatch/v2 v2.4.0 h1:Ci3iUJyx9UeRx7CeFN8ARgGbkESwJK+KB9lLcWxY/Zw= +gomodules.xyz/jsonpatch/v2 v2.4.0/go.mod h1:AH3dM2RI6uoBZxn3LVrfvJ3E0/9dG4cSrbuBJT4moAY= +gonum.org/v1/gonum v0.0.0-20180816165407-929014505bf4/go.mod h1:Y+Yx5eoAFn32cQvJDxZx5Dpnq+c3wtXuadVZAcxbbBo= +gonum.org/v1/gonum v0.0.0-20181121035319-3f7ecaa7e8ca/go.mod h1:Y+Yx5eoAFn32cQvJDxZx5Dpnq+c3wtXuadVZAcxbbBo= +gonum.org/v1/gonum v0.6.0/go.mod h1:9mxDZsDKxgMAuccQkewq682L+0eCu4dCN2yonUJTCLU= +gonum.org/v1/netlib v0.0.0-20181029234149-ec6d1f5cefe6/go.mod h1:wa6Ws7BG/ESfp6dHfk7C6KdzKA7wR7u/rKwOGE66zvw= +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= +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.19.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/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= +google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= +google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM= +google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc= +google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg= +google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= +google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8= +google.golang.org/api v0.41.0/go.mod h1:RkxM5lITDfTzmyKFPt+wGrCJbVfniCr2ool8kTBzRTU= +google.golang.org/api v0.43.0/go.mod h1:nQsDGjRXMo4lvh5hP0TKqF244gqhGcr/YSIykhUk/94= +google.golang.org/api v0.47.0/go.mod h1:Wbvgpq1HddcWVtzsVLyfLp8lDg6AA241LmgIL59tHXo= +google.golang.org/api v0.48.0/go.mod h1:71Pr1vy+TAZRPkPs/xlCf5SsU8WjuAWv1Pfjbtukyy4= +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/appengine v1.6.7/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-20190716160619-c506a9f90610/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= +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-20190927181202-20e1ac93f88c/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-20200108215221-bd8f9a0ef82f/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-20200228133532-8c2c7df3a383/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200420144010-e5e8543f8aeb/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U= +google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= +google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA= +google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210222152913-aa3ee6e6a81c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210303154014-9728d6b83eeb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210310155132-4ce2db91004e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= +google.golang.org/genproto v0.0.0-20210513213006-bf773b8c8384/go.mod h1:P3QM42oQyzQSnHPnZ/vqoCdDmzH28fzWByN9asMeM8A= +google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= +google.golang.org/genproto v0.0.0-20210604141403-392c879c8b08/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= +google.golang.org/genproto v0.0.0-20230822172742-b8732ec3820d h1:VBu5YqKPv6XiJ199exd8Br+Aetz+o08F+PLMnwJQHAY= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240701130421-f6361c86f094 h1:BwIjyKYGsK9dMCBOorzRri8MQwmi7mT9rGHsCEinZkA= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240701130421-f6361c86f094/go.mod h1:Ue6ibwXGpU+dqIcODieyLOcgj7z8+IcskoNIgZxtrFY= +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.24.0/go.mod h1:XDChyiUovWa60DnaeDeZmSW86xtLtjtZbwvSiRnRtcA= +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.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKal+60= +google.golang.org/grpc v1.29.0/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= +google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= +google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= +google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= +google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= +google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= +google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= +google.golang.org/grpc v1.36.1/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= +google.golang.org/grpc v1.37.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= +google.golang.org/grpc v1.37.1/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= +google.golang.org/grpc v1.38.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= +google.golang.org/grpc v1.65.0 h1:bs/cUb4lp1G5iImFFd3u5ixQzweKizoZJAwBNLR42lc= +google.golang.org/grpc v1.65.0/go.mod h1:WgYC2ypjlB0EiQi6wdKixMqukr6lBc0Vo+oOgjrM5ZQ= +google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= +google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= +google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= +google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= +google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= +google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= +google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= +google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= +google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= +google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= +google.golang.org/protobuf v1.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg= +google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw= +gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= +gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= +gopkg.in/evanphx/json-patch.v4 v4.12.0 h1:n6jtcsulIzXPJaxegRbvFNNrZDjbij7ny3gmSPG+6V4= +gopkg.in/evanphx/json-patch.v4 v4.12.0/go.mod h1:p8EYWUEYMpynmqDbY58zCKCFZw8pRWMG4EsWvDvM72M= +gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= +gopkg.in/fsnotify/fsnotify.v1 v1.4.7 h1:XNNYLJHt73EyYiCZi6+xjupS9CpvmiDgjPTAjrBlQbo= +gopkg.in/fsnotify/fsnotify.v1 v1.4.7/go.mod h1:Fyux9zXlo4rWoMSIzpn9fDAYjalPqJ/K1qJ27s+7ltE= +gopkg.in/gcfg.v1 v1.2.3/go.mod h1:yesOnuUOFQAhST5vPY4nbZsb/huCgGGXlipJsBn0b3o= +gopkg.in/inf.v0 v0.9.1 h1:73M5CoZyi3ZLMOyDlQh031Cx6N9NDJ2Vvfl76EDAgDc= +gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= +gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k= +gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo= +gopkg.in/square/go-jose.v2 v2.2.2/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/validator.v2 v2.0.1 h1:xF0KWyGWXm/LM2G1TrEjqOu4pa6coO9AlWSf3msVfDY= +gopkg.in/validator.v2 v2.0.1/go.mod h1:lIUZBlB3Im4s/eYp39Ry/wkR02yOPhZ9IwIRBjuPuG8= +gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= +gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= +gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20200605160147-a5ece683394c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20200615113413-eeeca48fe776/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/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= +gotest.tools/v3 v3.0.2/go.mod h1:3SzNCllyD9/Y+b5r9JIKQ474KzkZyqLqEfYqMsX94Bk= +gotest.tools/v3 v3.0.3/go.mod h1:Z7Lb0S5l+klDB31fvDQX8ss/FlKDxtlFlw3Oa8Ymbl8= +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.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= +k8s.io/api v0.17.5/go.mod h1:0zV5/ungglgy2Rlm3QK8fbxkXVs+BSJWpJP/+8gUVLY= +k8s.io/api v0.18.3/go.mod h1:UOaMwERbqJMfeeeHc8XJKawj4P9TgDRnViIqqBeH2QA= +k8s.io/api v0.21.1/go.mod h1:FstGROTmsSHBarKc8bylzXih8BLNYTiS3TZcsoEDg2s= +k8s.io/api v0.31.1 h1:Xe1hX/fPW3PXYYv8BlozYqw63ytA92snr96zMW9gWTU= +k8s.io/api v0.31.1/go.mod h1:sbN1g6eY6XVLeqNsZGLnI5FwVseTrZX7Fv3O26rhAaI= +k8s.io/apiextensions-apiserver v0.18.3/go.mod h1:TMsNGs7DYpMXd+8MOCX8KzPOCx8fnZMoIGB24m03+JE= +k8s.io/apiextensions-apiserver v0.31.0 h1:fZgCVhGwsclj3qCw1buVXCV6khjRzKC5eCFt24kyLSk= +k8s.io/apiextensions-apiserver v0.31.0/go.mod h1:b9aMDEYaEe5sdK+1T0KU78ApR/5ZVp4i56VacZYEHxk= +k8s.io/apimachinery v0.17.5/go.mod h1:ioIo1G/a+uONV7Tv+ZmCbMG1/a3kVw5YcDdncd8ugQ0= +k8s.io/apimachinery v0.18.3/go.mod h1:OaXp26zu/5J7p0f92ASynJa1pZo06YlV9fG7BoWbCko= +k8s.io/apimachinery v0.21.1/go.mod h1:jbreFvJo3ov9rj7eWT7+sYiRx+qZuCYXwWT1bcDswPY= +k8s.io/apimachinery v0.31.1 h1:mhcUBbj7KUjaVhyXILglcVjuS4nYXiwC+KKFBgIVy7U= +k8s.io/apimachinery v0.31.1/go.mod h1:rsPdaZJfTfLsNJSQzNHQvYoTmxhoOEofxtOsF3rtsMo= +k8s.io/apiserver v0.18.3/go.mod h1:tHQRmthRPLUtwqsOnJJMoI8SW3lnoReZeE861lH8vUw= +k8s.io/autoscaler/vertical-pod-autoscaler v0.10.1-0.20240925135200-1c35e1186735 h1:BcH6DuuRX1rkYmujoqb1YeC8Vv1O4J9aLPt3Ef8+TSc= +k8s.io/autoscaler/vertical-pod-autoscaler v0.10.1-0.20240925135200-1c35e1186735/go.mod h1:9ywHbt0kTrLyeNGgTNm7WEns34PmBMEr+9bDKTxW6wQ= +k8s.io/client-go v0.17.5/go.mod h1:S8uZpBpjJJdEH/fEyxcqg7Rn0P5jH+ilkgBHjriSmNo= +k8s.io/client-go v0.18.3/go.mod h1:4a/dpQEvzAhT1BbuWW09qvIaGw6Gbu1gZYiQZIi1DMw= +k8s.io/client-go v0.21.1/go.mod h1:/kEw4RgW+3xnBGzvp9IWxKSNA+lXn3A7AuH3gdOAzLs= +k8s.io/client-go v0.31.1 h1:f0ugtWSbWpxHR7sjVpQwuvw9a3ZKLXX0u0itkFXufb0= +k8s.io/client-go v0.31.1/go.mod h1:sKI8871MJN2OyeqRlmA4W4KM9KBdBUpDLu/43eGemCg= +k8s.io/code-generator v0.18.3/go.mod h1:TgNEVx9hCyPGpdtCWA34olQYLkh3ok9ar7XfSsr8b6c= +k8s.io/component-base v0.18.3/go.mod h1:bp5GzGR0aGkYEfTj+eTY0AN/vXTgkJdQXjNTTVUaa3k= +k8s.io/component-base v0.31.1 h1:UpOepcrX3rQ3ab5NB6g5iP0tvsgJWzxTyAo20sgYSy8= +k8s.io/component-base v0.31.1/go.mod h1:WGeaw7t/kTsqpVTaCoVEtillbqAhF2/JgvO0LDOMa0w= +k8s.io/gengo v0.0.0-20190128074634-0689ccc1d7d6/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0= +k8s.io/gengo v0.0.0-20200114144118-36b2048a9120/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0= +k8s.io/gengo v0.0.0-20200413195148-3a45101e95ac/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0= +k8s.io/klog v0.0.0-20181102134211-b9b56d5dfc92/go.mod h1:Gq+BEi5rUBO/HRz0bTSXDUcqjScdoY3a9IHpCEIOOfk= +k8s.io/klog v0.3.0/go.mod h1:Gq+BEi5rUBO/HRz0bTSXDUcqjScdoY3a9IHpCEIOOfk= +k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I= +k8s.io/klog/v2 v2.0.0/go.mod h1:PBfzABfn139FHAV07az/IF9Wp1bkk3vpT2XSJ76fSDE= +k8s.io/klog/v2 v2.8.0/go.mod h1:hy9LJ/NvuK+iVyP4Ehqva4HxZG/oXyIS3n3Jmire4Ec= +k8s.io/klog/v2 v2.9.0/go.mod h1:hy9LJ/NvuK+iVyP4Ehqva4HxZG/oXyIS3n3Jmire4Ec= +k8s.io/klog/v2 v2.130.1 h1:n9Xl7H1Xvksem4KFG4PYbdQCQxqc/tTUyrgXaOhHSzk= +k8s.io/klog/v2 v2.130.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE= +k8s.io/kube-openapi v0.0.0-20200316234421-82d701f24f9d/go.mod h1:F+5wygcW0wmRTnM3cOgIqGivxkwSWIWT5YdsDbeAOaU= +k8s.io/kube-openapi v0.0.0-20200410145947-61e04a5be9a6/go.mod h1:GRQhZsXIAJ1xR0C9bd8UpWHZ5plfAS9fzPjJuQ6JL3E= +k8s.io/kube-openapi v0.0.0-20210305001622-591a79e4bda7/go.mod h1:wXW5VT87nVfh/iLV8FpR2uDvrFyomxbtb1KivDbvPTE= +k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 h1:BZqlfIlq5YbRMFko6/PM7FjZpUb45WallggurYhKGag= +k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340/go.mod h1:yD4MZYeKMBwQKVht279WycxKyM84kkAx2DPrTXaeb98= +k8s.io/kubectl v0.31.0 h1:kANwAAPVY02r4U4jARP/C+Q1sssCcN/1p9Nk+7BQKVg= +k8s.io/kubectl v0.31.0/go.mod h1:pB47hhFypGsaHAPjlwrNbvhXgmuAr01ZBvAIIUaI8d4= +k8s.io/utils v0.0.0-20191114184206-e782cd3c129f/go.mod h1:sZAwmy6armz5eXlNoLmJcl4F1QuKu7sr+mFQ0byX7Ew= +k8s.io/utils v0.0.0-20200324210504-a9aa75ae1b89/go.mod h1:sZAwmy6armz5eXlNoLmJcl4F1QuKu7sr+mFQ0byX7Ew= +k8s.io/utils v0.0.0-20200414100711-2df71ebbae66/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA= +k8s.io/utils v0.0.0-20201110183641-67b214c5f920/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA= +k8s.io/utils v0.0.0-20240711033017-18e509b52bc8 h1:pUdcCO1Lk/tbT5ztQWOBi5HBgbBP1J8+AsQnQCKsi8A= +k8s.io/utils v0.0.0-20240711033017-18e509b52bc8/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0= +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/apiserver-network-proxy/konnectivity-client v0.0.7/go.mod h1:PHgbrJT7lCHcxMU+mDHEm+nx46H4zuuHZkDP6icnhu0= +sigs.k8s.io/controller-runtime v0.19.0 h1:nWVM7aq+Il2ABxwiCizrVDSlmDcshi9llbaFbC0ji/Q= +sigs.k8s.io/controller-runtime v0.19.0/go.mod h1:iRmWllt8IlaLjvTTDLhRBXIEtkCK6hwVBJJsYS9Ajf4= +sigs.k8s.io/e2e-framework v0.5.0 h1:YLhk8R7EHuTFQAe6Fxy5eBzn5Vb+yamR5u8MH1Rq3cE= +sigs.k8s.io/e2e-framework v0.5.0/go.mod h1:jJSH8u2RNmruekUZgHAtmRjb5Wj67GErli9UjLSY7Zc= +sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd h1:EDPBXCAspyGV4jQlpZSudPeMmr1bNJefnuqLsRAsHZo= +sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd/go.mod h1:B8JuhiUyNFVKdsE8h686QcCxMaH6HrOAZj4vswFpcB0= +sigs.k8s.io/structured-merge-diff/v2 v2.0.1/go.mod h1:Wb7vfKAodbKgf6tn1Kl0VvGj7mRH6DGaRcixXEJXTsE= +sigs.k8s.io/structured-merge-diff/v3 v3.0.0-20200116222232-67a7b8c61874/go.mod h1:PlARxl6Hbt/+BC80dRLi1qAmnMqwqDg62YvvVkZjemw= +sigs.k8s.io/structured-merge-diff/v3 v3.0.0/go.mod h1:PlARxl6Hbt/+BC80dRLi1qAmnMqwqDg62YvvVkZjemw= +sigs.k8s.io/structured-merge-diff/v4 v4.0.2/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw= +sigs.k8s.io/structured-merge-diff/v4 v4.1.0/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw= +sigs.k8s.io/structured-merge-diff/v4 v4.4.1 h1:150L+0vs/8DA78h1u02ooW1/fFq/Lwr+sGiqlzvrtq4= +sigs.k8s.io/structured-merge-diff/v4 v4.4.1/go.mod h1:N8hJocpFajUSSeSJ9bOZ77VzejKZaXsTtZo4/u7Io08= +sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o= +sigs.k8s.io/yaml v1.2.0/go.mod h1:yfXDCHCao9+ENCvLSE62v9VSji2MKu5jeNfTrofGhJc= +sigs.k8s.io/yaml v1.4.0 h1:Mk1wCc2gy/F0THH0TAp1QYyJNzRm2KCLy3o5ASXVI5E= +sigs.k8s.io/yaml v1.4.0/go.mod h1:Ejl7/uTz7PSA4eKMyQCUTnhZYNmLIl+5c2lQPGR2BPY= +sourcegraph.com/sourcegraph/appdash v0.0.0-20190731080439-ebfcffb1b5c0/go.mod h1:hI742Nqp5OhwiqlzhgfbWU4mW4yO10fP+LoT9WOswdU= diff --git a/helm-chart/observability-operator/.helmignore b/helm-chart/observability-operator/.helmignore new file mode 100644 index 0000000..f0c1319 --- /dev/null +++ b/helm-chart/observability-operator/.helmignore @@ -0,0 +1,21 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*~ +# Various IDEs +.project +.idea/ +*.tmproj diff --git a/helm-chart/observability-operator/Chart.yaml b/helm-chart/observability-operator/Chart.yaml new file mode 100644 index 0000000..57ae9b9 --- /dev/null +++ b/helm-chart/observability-operator/Chart.yaml @@ -0,0 +1,7 @@ +apiVersion: v1 +description: A Helm chart for Observability Operator +name: observability-operator +version: 1.1.3 +maintainers: +- name: Adevinta + email: gp.gt.cpr@adevinta.com diff --git a/helm-chart/observability-operator/templates/_helpers.tpl b/helm-chart/observability-operator/templates/_helpers.tpl new file mode 100644 index 0000000..862fd60 --- /dev/null +++ b/helm-chart/observability-operator/templates/_helpers.tpl @@ -0,0 +1,24 @@ +{{/* vim: set filetype=mustache: */}} +{{/* +Expand the name of the chart. +*/}} +{{- define "name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 24 | trimSuffix "-" -}} +{{- end -}} + +{{/* +Create a default fully qualified app name. +We truncate at 24 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +*/}} +{{- define "fullname" -}} +{{- $name := default .Chart.Name .Values.nameOverride -}} +{{- printf "%s-%s" .Release.Name $name | trunc 24 | trimSuffix "-" -}} +{{- end -}} + +{{- define "deployment.apiVersion" -}} +{{- if ge .Capabilities.KubeVersion.Minor "16" -}} +"apps/v1" +{{- else -}} +"extensions/v1beta1" +{{- end -}} +{{- end -}} diff --git a/helm-chart/observability-operator/templates/deployment.yaml b/helm-chart/observability-operator/templates/deployment.yaml new file mode 100644 index 0000000..96bf581 --- /dev/null +++ b/helm-chart/observability-operator/templates/deployment.yaml @@ -0,0 +1,91 @@ +apiVersion: {{ include "deployment.apiVersion" $ }} +kind: Deployment +metadata: + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} + labels: + chart: "{{ .Chart.Name }}" +spec: + replicas: {{ .Values.replicaCount }} + selector: + matchLabels: + app: {{ .Release.Name }} + template: + metadata: + labels: + app: {{ .Release.Name }} + annotations: + {{ if $.Values.roleARN }} + iam.amazonaws.com/role: {{ $.Values.roleARN | quote }} + {{- end }} + spec: + serviceAccountName: {{ .Release.Name }} + automountServiceAccountToken: true + containers: + - name: {{ .Chart.Name }} + image: {{ .Values.image.registry }}/{{ .Values.image.repository }}:{{ .Values.image.tag }} + imagePullPolicy: {{ .Values.pullPolicy }} + args: + - -cluster-name={{ .Values.clusterName }} + - -cluster-region={{ .Values.region }} + {{- if .Values.clusterDomain }} + - -cluster-domain={{ .Values.clusterDomain }} + {{- end }} + {{- with .Values.filtering.excludeLabels }} + - -exclude-apps-label={{ join "," . }} + {{- end }} + {{- with .Values.filtering.excludeNamespaces }} + - -exclude-namespaces-name={{ join "," . }} + {{- end }} + {{- with .Values.exclusionLabelSelectors.workload }} + - -exclude-workload-selector={{ join "," . }} + {{- end }} + {{- with .Values.exclusionLabelSelectors.namespace }} + - -exclude-namespace-selector={{ join "," . }} + {{- end }} + {{- if and .Values.prometheusDockerImage.registry .Values.prometheusDockerImage.repository .Values.prometheusDockerImage.tag }} + - -prometheus-docker-image={{ .Values.prometheusDockerImage.registry }}/{{ .Values.prometheusDockerImage.repository }} + - -prometheus-docker-tag={{ .Values.prometheusDockerImage.tag }} + {{- end }} + - -metrics-remote-write-to-grafana-cloud={{ .Values.enableGrafanaCloud }} + - -enable-vpa={{ .Values.enableVpa }} + - -prometheus-namespace={{ .Values.namespaces.prometheusNamespace.name }} + {{- with .Values.prometheusNodeSelectorTarget }} + - -prometheus-node-selector-target={{ . }} + {{- end }} + - -traces-namespace={{ .Values.namespaces.tracesNamespace.name }} + - -grafana-cloud-client-use-cache={{ .Values.grafanacloud.client.useCache }} + - -grafana-cloud-metrics-credentials={{ .Values.credentials.grafana_cloud_metrics_token.secretName }} + {{- if .Values.grafanacloud.configmap }} + - -logs-fluentd-loki-configmap-namespace={{ .Values.grafanacloud.configmap.namespace }} + - -logs-fluentd-loki-configmap-name={{ .Values.grafanacloud.configmap.name }} + - -logs-fluentd-loki-configmap-key={{ .Values.grafanacloud.configmap.lokikey }} + {{- end }} + {{- with .Values.prometheus.priorityClassName }} + - -prometheus-pod-priority-classname={{ . }} + {{- end }} + - -prometheus-service-account-name={{ .Values.prometheus.serviceAccount.name | default (printf "%s-prometheus" .Release.Name) }} + {{- $config := .Values.prometheus.extraScrapingConfiguration }} + {{- if and $config $config.monitoringTarget $config.monitoringTarget.name }} + - -prometheus-monitoring-target-name={{ $config.monitoringTarget.name }} + {{- end }} + {{- $config := .Values.prometheus }} + {{- if and $config $config.externalLabels }} + - -prometheus-extra-external-labels={{ .Values.prometheus.externalLabels }} + {{- end }} + env: + - name: GRAFANA_CLOUD_TOKEN + valueFrom: + secretKeyRef: + name: {{ .Values.credentials.GRAFANA_CLOUD_TOKEN.secretName }} + key: {{ .Values.credentials.GRAFANA_CLOUD_TOKEN.secretKey }} + - name: GRAFANA_CLOUD_TRACES_TOKEN + valueFrom: + secretKeyRef: + name: {{ .Values.credentials.GRAFANA_CLOUD_TRACES_TOKEN.secretName }} + key: {{ .Values.credentials.GRAFANA_CLOUD_TRACES_TOKEN.secretKey }} + ports: + - containerPort: {{ .Values.service.internalPort }} + name: prometheus + resources: +{{ toYaml .Values.resources | indent 12 }} diff --git a/helm-chart/observability-operator/templates/prometheus-tenant-rbac.yaml b/helm-chart/observability-operator/templates/prometheus-tenant-rbac.yaml new file mode 100644 index 0000000..a0aa98b --- /dev/null +++ b/helm-chart/observability-operator/templates/prometheus-tenant-rbac.yaml @@ -0,0 +1,40 @@ +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ .Release.Name }}-prometheus + labels: + app: {{ .Release.Name }}-prometheus +rules: +# This permission are not in the kube-prometheus repo +# they're grabbed from https://github.com/prometheus/prometheus/blob/master/documentation/examples/rbac-setup.yml +- apiGroups: [""] + resources: + - services + - endpoints + - pods + verbs: ["get", "list", "watch"] +- apiGroups: ["discovery.k8s.io"] + resources: + - endpointslices + verbs: ["get", "list", "watch"] +- nonResourceURLs: ["/metrics", "/metrics/cadvisor"] + verbs: ["get"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ .Release.Name }}-prometheus-scraping-rolebinding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-prometheus +subjects: +- kind: ServiceAccount + name: prometheus-tenant + namespace: {{ .Values.namespaces.prometheusNamespace.name }} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: prometheus-tenant + namespace: {{ .Values.namespaces.prometheusNamespace.name }} diff --git a/helm-chart/observability-operator/templates/rbac.yaml b/helm-chart/observability-operator/templates/rbac.yaml new file mode 100644 index 0000000..8ed98e7 --- /dev/null +++ b/helm-chart/observability-operator/templates/rbac.yaml @@ -0,0 +1,207 @@ +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ .Release.Name }}-manager-role +rules: +- apiGroups: ["monitoring.coreos.com"] + resources: ["prometheuses","servicemonitors","prometheusrules","podmonitors"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: ["extensions", "apps", "batch" ] + resources: ["*"] + verbs: ["get", "list", "watch"] +- apiGroups: ["" ] + resources: ["pods"] + verbs: ["get", "list", "watch"] +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["get", "delete", "patch", "list", "update", "watch"] +- apiGroups: ["extensions", "apps" ] + resources: ["deployments/status"] + verbs: ["get", "patch", "update", "list"] +- apiGroups: ["core", ""] + resources: ["services"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: ["core", ""] + resources: ["secrets"] + verbs: ["create", "get", "list", "patch", "update", "watch"] +- apiGroups: ["core", ""] + resources: ["services/status"] + verbs: ["get", "patch", "update", "list"] +- apiGroups: ["extensions", "networking.k8s.io"] + resources: ["ingresses"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: ["extensions" ] + resources: ["ingresses/status"] + verbs: ["get", "patch", "update", "list"] +- apiGroups: [autoscaling.k8s.io] + resources: ["verticalpodautoscalers"] + verbs: ["get", "create", "delete", "update", "list", "watch"] +- apiGroups: ["argoproj.io"] + resources: ["rollouts"] + verbs: ["get", "list", "watch"] +- apiGroups: [""] + resources: ["events"] + verbs: ["get", "list", "watch", "create", "patch"] +- apiGroups: [""] + resources: ["namespaces"] + verbs: ["get", "list"] + +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ .Release.Name }}-proxy-role +rules: +- apiGroups: ["authentication.k8s.io"] + resources: ["tokenreviews"] + verbs: ["create"] +- apiGroups: ["authorization.k8s.io"] + resources: ["subjectaccessreviews"] + verbs: ["create"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: {{ .Release.Name }}-metrics-reader +rules: +- nonResourceURLs: ["/metrics"] + verbs: ["get", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ .Release.Name }}-leader-election-role + namespace: {{ .Release.Namespace }} +rules: +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: [""] + resources: ["configmaps/status"] + verbs: ["get", "update", "patch", "list"] +- apiGroups: [""] + resources: ["events"] + verbs: ["create", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ .Release.Name }}-alloy-management + namespace: {{ .Values.namespaces.tracesNamespace.name }} +rules: +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: [""] + resources: ["configmaps/status"] + verbs: ["get", "update", "patch", "list"] +- apiGroups: [""] + resources: ["events"] + verbs: ["create", "list"] +- apiGroups: ["core", ""] + resources: ["secrets"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: ["core", ""] + resources: ["services"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: ["apps"] + resources: ["deployments"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +- apiGroups: ["networking.k8s.io"] + resources: ["networkpolicies"] + verbs: ["create", "delete", "get", "list", "patch", "update", "watch"] +{{- if .Values.grafanacloud.configmap }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: Role +metadata: + name: {{ .Release.Name }}-loki-fluentd-configmap-creation + namespace: {{ .Values.grafanacloud.configmap.namespace }} +rules: +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["create"] +- apiGroups: [""] + resources: ["configmaps"] + verbs: ["get", "delete", "patch", "list", "update", "watch"] + resourceNames: + - {{ .Values.grafanacloud.configmap.name }} +- apiGroups: [""] + resources: ["configmaps/status"] + verbs: ["get", "update", "patch", "list"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ .Release.Name }}-loki-fluentd-configmap-creation + namespace: {{ .Values.grafanacloud.configmap.namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ .Release.Name }}-loki-fluentd-configmap-creation +subjects: +- kind: ServiceAccount + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} +{{- end }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ .Release.Name }}-leader-election-rolebinding + namespace: {{ .Release.Namespace }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ .Release.Name }}-leader-election-role +subjects: +- kind: ServiceAccount + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: RoleBinding +metadata: + name: {{ .Release.Name }}-alloy-management + namespace: {{ .Values.namespaces.tracesNamespace.name }} +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: Role + name: {{ .Release.Name }}-alloy-management +subjects: +- kind: ServiceAccount + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ .Release.Name }}-manager-rolebinding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-manager-role +subjects: +- kind: ServiceAccount + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: {{ .Release.Name }}-proxy-rolebinding +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: {{ .Release.Name }}-proxy-role +subjects: +- kind: ServiceAccount + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} +--- +apiVersion: v1 +kind: ServiceAccount +metadata: + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} diff --git a/helm-chart/observability-operator/templates/vpa.yaml b/helm-chart/observability-operator/templates/vpa.yaml new file mode 100644 index 0000000..b0deb23 --- /dev/null +++ b/helm-chart/observability-operator/templates/vpa.yaml @@ -0,0 +1,13 @@ +{{ if $.Values.enableSelfVpa }} +apiVersion: autoscaling.k8s.io/v1 +kind: VerticalPodAutoscaler +metadata: + name: {{ .Release.Name }} + namespace: {{ .Release.Namespace }} +spec: + targetRef: + kind: Deployment + name: {{ .Release.Name }} + updatePolicy: + updateMode: Auto +{{ end }} diff --git a/helm-chart/observability-operator/values-test-full.yaml b/helm-chart/observability-operator/values-test-full.yaml new file mode 100644 index 0000000..164ce4d --- /dev/null +++ b/helm-chart/observability-operator/values-test-full.yaml @@ -0,0 +1,120 @@ +# Default values for observability-operator. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. +replicaCount: 1 +image: + registry: ghcr.io + repository: adevinta/observability-operator + tag: latest + +pullPolicy: IfNotPresent +region: eu-west-1 +clusterName: CHANGEME + +# Domain name of the cluster. +# Used for: +# - annotations and label keys to configure and maintain +# relations between objects managed by this operator +# - determine the internal URL Prometheus statefulsets should expose +clusterDomain: CHANGEME + +roleARN: CHANGEME +enableGrafanaCloud: true +enableVpa: false + +prometheus: +# The service account to be used by the Prometheus instances, which provides the necessary permissions to scrape workloads +# if name is empty, the name will be dynamically generated based on the chart release name. If the user has already an existing sa then create:false and name must be filled. + serviceAccount: + name: "prometheus-tenant" +# Priority class name to be used by the Prometheus instances + priorityClassName: "" + + +# To enable metrics collection, the following values must be set. +# If any of these are missing, the operator will not fetch metrics. +prometheusDockerImage: + registry: "quay.io" + repository: "prometheus/prometheus" + tag: "v2.43.0" + +# Defines which nodes can host Prometheus instances +# prometheusNodeSelectorTarget: "CHANGEME" + +service: + internalPort: 8080 + +enableSelfVpa: true + +# Credentials holds the references to the Secret objects that hold the +# credentials required to interact with GrafanaCloud +credentials: + # GRAFANA_CLOUD_TOKEN is an expected Environment Variable to be set + # in the operator holding an API key for the Grafana.com API, so it + # can discover existing stacks and access values + GRAFANA_CLOUD_TOKEN: + # secretName - name of the secret that holds API key for grafana.com + # and the token for the Cloud Access policy that allows to write + # traces to GrafanaCloud + secretName: "observability-operator-grafana-cloud-credentials" + # secretKey - the key inside the secret holding the value + secretKey: "grafana-cloud-api-key" + # GRAFANA_CLOUD_TRACES_TOKEN is an expected Environment Variable to + # be set in the operator holding a Cloud Access Policy token to + # write the traces endpoint of any stack in the organization + GRAFANA_CLOUD_TRACES_TOKEN: + # secretName - name of the secret that holds API key for grafana.com + # and the token for the Cloud Access policy that allows to write + # traces to GrafanaCloud + secretName: "observability-operator-grafana-cloud-credentials" + # secretKey - the key inside the secret holding the value + secretKey: "grafana-cloud-traces-token" + grafana_cloud_metrics_token: + # secretName - name of the secret that holds API that + # allows to write prometheus data to GrafanaCloud, stored in the + # Prometheus namespace + secretName: "observability-operator-grafana-cloud-credentials" + # secretKey - the key inside the secret holding the value + # TODO: still hardcoded in codebase + secretKey: "grafana-cloud-api-key" + +# Namespaces to be used by the operator and its deployments +namespaces: + # tracesNamespace - the namespace to hold Alloy workers to act as + # per-tenant OTel collectors for traces + tracesNamespace: + # name - name of the namespace + name: "observability" + # prometheusNamespace - the namespace to hold Prometheus per-tenant + # to scrape workloads and forward them to the destination + prometheusNamespace: + # name - name of the namespace + name: "platform-services" + +resources: + limits: + cpu: 100m + memory: 128Mi + requests: + cpu: 100m + memory: 128Mi +grafanacloud: + configmap: {} + # namespace: platform-services + # name: grafana-cloud-config + # lokikey: loki + client: + useCache: false + +filtering: + excludeLabels: [] + excludeNamespaces: [] + +# Use as described here https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors +# to select the objects that should be filtered out by the operator, multiple selectors can be used and will be combined +# All selectors must be matched for the object to be skipped +exclusionLabelSelectors: + # list of string selectors to match against deployments, replicasets, statefulsets, daemonsets, etc + workload: [] + # list of string selectors to match against namespaces + namespace: [] diff --git a/helm-chart/observability-operator/values.yaml b/helm-chart/observability-operator/values.yaml new file mode 100644 index 0000000..f122dc8 --- /dev/null +++ b/helm-chart/observability-operator/values.yaml @@ -0,0 +1,125 @@ +# Default values for observability-operator. +# This is a YAML-formatted file. +# Declare variables to be passed into your templates. +replicaCount: 1 +image: + registry: ghcr.io + repository: adevinta/observability-operator + tag: latest + +pullPolicy: IfNotPresent +region: eu-west-1 +clusterName: CHANGEME + +# Domain name of the cluster. +# Used for: +# - annotations and label keys to configure and maintain +# relations between objects managed by this operator +# - determine the internal URL Prometheus statefulsets should expose +clusterDomain: CHANGEME + +roleARN: CHANGEME +enableGrafanaCloud: true +enableVpa: false + +prometheus: +# The service account to be used by the Prometheus instances, which provides the necessary permissions to scrape workloads +# if name is empty, the name will be dynamically generated based on the chart release name. If the user has already an existing sa then create:false and name must be filled. + serviceAccount: + name: "prometheus-tenant" + # Priority class name to be used by the Prometheus instances + priorityClassName: "" +# externalLabels: "key1:value1,key2:value2" + + # Prometheus configuration for additional scraping targets. + # extraScrapingConfiguration: + # monitoringTarget: + # name: "cluster-ingress-prometheus-metrics" + +# To enable metrics collection, the following values must be set. +# If any of these are missing, the operator will not fetch metrics. +prometheusDockerImage: + registry: "quay.io" + repository: "prometheus/prometheus" + tag: "v2.43.0" + +# Defines which nodes can host Prometheus instances +# prometheusNodeSelectorTarget: "CHANGEME" + +service: + internalPort: 8080 + +enableSelfVpa: true + +# Credentials holds the references to the Secret objects that hold the +# credentials required to interact with GrafanaCloud +credentials: + # GRAFANA_CLOUD_TOKEN is an expected Environment Variable to be set + # in the operator holding an API key for the Grafana.com API, so it + # can discover existing stacks and access values + GRAFANA_CLOUD_TOKEN: + # secretName - name of the secret that holds API key for grafana.com + # and the token for the Cloud Access policy that allows to write + # traces to GrafanaCloud + secretName: "observability-operator-grafana-cloud-credentials" + # secretKey - the key inside the secret holding the value + secretKey: "grafana-cloud-api-key" + # GRAFANA_CLOUD_TRACES_TOKEN is an expected Environment Variable to + # be set in the operator holding a Cloud Access Policy token to + # write the traces endpoint of any stack in the organization + GRAFANA_CLOUD_TRACES_TOKEN: + # secretName - name of the secret that holds API key for grafana.com + # and the token for the Cloud Access policy that allows to write + # traces to GrafanaCloud + secretName: "observability-operator-grafana-cloud-credentials" + # secretKey - the key inside the secret holding the value + secretKey: "grafana-cloud-traces-token" + grafana_cloud_metrics_token: + # secretName - name of the secret that holds API that + # allows to write prometheus data to GrafanaCloud, stored in the + # Prometheus namespace + secretName: "observability-operator-grafana-cloud-credentials" + # secretKey - the key inside the secret holding the value + # TODO: still hardcoded in codebase + secretKey: "grafana-cloud-api-key" + +# Namespaces to be used by the operator and its deployments +namespaces: + # tracesNamespace - the namespace to hold Alloy workers to act as + # per-tenant OTel collectors for traces + tracesNamespace: + # name - name of the namespace + name: "observability" + # prometheusNamespace - the namespace to hold Prometheus per-tenant + # to scrape workloads and forward them to the destination + prometheusNamespace: + # name - name of the namespace + name: "platform-services" + +resources: + limits: + cpu: 100m + memory: 128Mi + requests: + cpu: 100m + memory: 128Mi +grafanacloud: + configmap: {} + # namespace: platform-services + # name: grafana-cloud-config + # lokikey: loki + client: + useCache: false + +filtering: + excludeLabels: [] + excludeNamespaces: [] + +# Use as described here https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors +# to select the objects that should be filtered out by the operator, multiple selectors can be used and will be combined +# All selectors must be matched for the object to be skipped +exclusionLabelSelectors: + # list of string selectors to match against deployments, replicasets, statefulsets, daemonsets, etc + workload: [] + # list of string selectors to match against namespaces + namespace: [] diff --git a/pkg/controllers/common.go b/pkg/controllers/common.go new file mode 100644 index 0000000..b193d1f --- /dev/null +++ b/pkg/controllers/common.go @@ -0,0 +1,230 @@ +package controllers + +import ( + "context" + "fmt" + "strings" + "time" + + "github.com/go-logr/logr" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + corev1 "k8s.io/api/core/v1" + apierrors "k8s.io/apimachinery/pkg/api/errors" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/labels" + "k8s.io/apimachinery/pkg/runtime" + types "k8s.io/apimachinery/pkg/types" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" + clientgoscheme "k8s.io/client-go/kubernetes/scheme" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" +) + +var Config *config = NewConfig("adevinta.com") + +type config struct { + // Destination GrafanaCloud stack selection + stackNameAnnotationKey string + // Namespace feature toggles + metricsLabelKey string + logsLabelKey string + tracesAnnotationKey string + // Advanced user settings - Prometheus RemoteWrite + referencedSecretAnnotationKeys string + remoteWriteAnnotationKey string + // Advanced user settings - Prometheus ingestion metrics rate limit + podSampleLimitAnnotation string + // Internal interfaces - PodMonitor + storageAnnotationKey string + alertManagerAnnotationKey string + accountAnnotationKey string + // This is expected at tenant namespace level + accountLabelKey string + // Interface with Prometheus Operator? + podMonitorAnnotationKey string + podMonitorLabelKey string + // Finalizers + podmonitorFinalizer string +} + +func NewConfig(baseDomain string) *config { + return &config{ + stackNameAnnotationKey: "grafanacloud." + baseDomain + "/stack-name", + metricsLabelKey: "grafanacloud." + baseDomain + "/metrics", + logsLabelKey: "grafanacloud." + baseDomain + "/logs", + tracesAnnotationKey: "grafanacloud." + baseDomain + "/traces", + referencedSecretAnnotationKeys: "monitoring." + baseDomain + "/referenced-secrets", + remoteWriteAnnotationKey: "metrics.monitoring." + baseDomain + "/remote-write", + podSampleLimitAnnotation: "monitor." + baseDomain + "/pod-sample-limit", + storageAnnotationKey: "monitor." + baseDomain + "/storage", + alertManagerAnnotationKey: "monitor." + baseDomain + "/alert-manager", + accountAnnotationKey: "monitor." + baseDomain + "/account", + accountLabelKey: "monitor." + baseDomain + "/account", + podMonitorAnnotationKey: baseDomain + "/podmonitor", + podMonitorLabelKey: baseDomain + "/podmonitor", + podmonitorFinalizer: "finalizer.podmonitor." + baseDomain, + } +} + +func getObjectName(prometheus metav1.Object) string { + return "prometheus-" + prometheus.GetName() +} + +func NewScheme() *runtime.Scheme { + scheme := runtime.NewScheme() + _ = clientgoscheme.AddToScheme(scheme) + + _ = monitoringv1.AddToScheme(scheme) + _ = vpav1.AddToScheme(scheme) + + return scheme +} + +func newPrometheusObjectDef(namespace, prometheusNamespace string) *monitoringv1.Prometheus { + return &monitoringv1.Prometheus{ + ObjectMeta: metav1.ObjectMeta{ + Name: namespace, + Namespace: prometheusNamespace, + }, + Spec: monitoringv1.PrometheusSpec{}, + } +} + +func isGrafanaCloudStorageEnabled(objectMeta metav1.ObjectMeta) bool { + value, ok := getStorageAnnotation(objectMeta.Annotations) + if ok { + availableStorages := strings.Split(value, ",") + return findInAnnotation(availableStorages, "grafanacloud") + } + return false +} + +func publishPrometheusErrEvent(client ctrlclient.Client, eventName, eventNamespace string, prometheus *monitoringv1.Prometheus, e error) error { + eventToPublish := &corev1.Event{ + ObjectMeta: metav1.ObjectMeta{ + Name: eventName, + Namespace: eventNamespace, + }, + ReportingController: "NamespaceController", + ReportingInstance: prometheus.Name, + InvolvedObject: corev1.ObjectReference{ + Kind: prometheus.Kind, + Namespace: prometheus.Namespace, + Name: prometheus.Name, + }, + LastTimestamp: metav1.NewTime(time.Now()), + Type: "Warning", + EventTime: metav1.NowMicro(), + Action: "ignore-relabeling", + Source: corev1.EventSource{ + Component: "grafana-cloud-operator", + }, + Reason: "invalid-remote-write", + } + + _, err := ctrl.CreateOrUpdate( + context.TODO(), + client, + eventToPublish, + func() error { + eventToPublish.Message = e.Error() + eventToPublish.Count = eventToPublish.Count + 1 + eventToPublish.LastTimestamp = metav1.NewTime(time.Now()) + eventToPublish.InvolvedObject = corev1.ObjectReference{ + Kind: prometheus.Kind, + Namespace: prometheus.Namespace, + Name: prometheus.Name, + } + eventToPublish.Type = "Warning" + if eventToPublish.EventTime.IsZero() { + eventToPublish.EventTime = metav1.NowMicro() + } + eventToPublish.Action = "ignore-relabeling" + eventToPublish.Source = corev1.EventSource{ + Component: "grafana-cloud-operator", + } + eventToPublish.Reason = "invalid-remote-write" + return nil + }, + ) + return err +} + +func getSecret(client ctrlclient.Client, secret types.NamespacedName) (*corev1.Secret, error) { + secretObject := corev1.Secret{} + + if err := client.Get(context.Background(), secret, &secretObject); err != nil { + return nil, err + } + + return &secretObject, nil +} + +func checkNamespaceHasActionableWorkloads(k8sClient ctrlclient.Client, log logr.Logger, namespace string, filters ...podFilter) ([]corev1.Pod, error) { + podList := &corev1.PodList{} + + ctx := context.Background() + if err := k8sClient.List(ctx, podList, ctrlclient.InNamespace(namespace)); err != nil { + log.Error(err, "unable to list pods in tenant namespace") + return []corev1.Pod{}, err + } + + out := podList.Items + for _, f := range filters { + out = f(out) + } + return out, nil +} + +type podFilter func(p []corev1.Pod) []corev1.Pod + +func filterBySelector(selector labels.Selector) podFilter { + return func(p []corev1.Pod) []corev1.Pod { + var filteredPods []corev1.Pod + for _, pod := range p { + if !selector.Matches(labels.Set(pod.Labels)) { + filteredPods = append(filteredPods, pod) + } + } + return filteredPods + } +} + +func excludePodsOnLabel(label, value string) podFilter { + return func(p []corev1.Pod) []corev1.Pod { + var filteredPods []corev1.Pod + for _, pod := range p { + if !strings.Contains(pod.Labels[label], value) { + filteredPods = append(filteredPods, pod) + } + } + return filteredPods + } +} + +func lookupGrafanaStacks(k8sclient client.Client, namespace string) ([]string, error) { + ns := corev1.Namespace{} + err := k8sclient.Get(context.Background(), client.ObjectKey{Name: namespace}, &ns) + if err != nil { + if apierrors.IsNotFound(err) { + return []string{}, nil + } + return []string{}, err + } + + stack, ok := ns.GetAnnotations()[Config.stackNameAnnotationKey] + if !ok { + return []string{}, fmt.Errorf("the annotation %s is not set on the namespace %s", Config.stackNameAnnotationKey, namespace) + } + + split := strings.Split(stack, ",") + + var trimmed []string + for _, part := range split { + trimmed = append(trimmed, strings.TrimSpace(part)) + } + + return trimmed, nil + +} diff --git a/pkg/controllers/files/config.alloy b/pkg/controllers/files/config.alloy new file mode 100644 index 0000000..8ddbc3f --- /dev/null +++ b/pkg/controllers/files/config.alloy @@ -0,0 +1,136 @@ +logging { + level = "info" + format = "json" +} + +tracing { + write_to = [otelcol.processor.transform.default.input] //TODO send to our grafana stack ? +} + +// TODO Metrics + +otelcol.receiver.otlp "default" { + grpc { + endpoint = "0.0.0.0:4317" + } + + http { + endpoint = "0.0.0.0:4318" + } + + output { + traces = [otelcol.processor.tail_sampling.default.input] + } +} + +// Tail sampling (WARNING this is a stateful component) +// https://grafana.com/docs/alloy/latest/reference/components/otelcol/otelcol.processor.tail_sampling/ +otelcol.processor.tail_sampling "default" { + // Total wait time from the start of a trace before making a sampling decision. + // Note that smaller time periods can potentially cause a decision to be made + // before the end of a trace has occurred. + // We keep the default value (to be optimized later if necessary) + decision_wait = "30s" + + // Determines the buffer size of the trace delete channel which is composed of trace ids. + // Increasing the number will increase the memory usage of the component + // while decreasing the number will lower the maximum amount of traces kept in memory + // We keep the default value (to be optimized later if necessary) + num_traces = 50000 + + // Determines the initial slice sizing of the current batch. + // A larger number will use more memory but be more efficient when adding traces to the batch + // We keep the default value (to be optimized later if necessary) + expected_new_traces_per_sec = 0 + + // We keep traces in error status + policy { + name = "sample-error-traces" + type = "status_code" + + status_code { + status_codes = ["ERROR"] + } + } + + // We keep a subset/sample of successful traces + policy { + name = "sample-probabilistic-successful" + type = "and" + + and { + and_sub_policy { + name = "successful-traces" + type = "status_code" + + status_code { + status_codes = ["OK", "UNSET"] + } + } + + and_sub_policy { + name = "sample-probabilistic" + type = "probabilistic" + + probabilistic { + sampling_percentage = 10 + } + } + } + } + + output { + traces = [otelcol.processor.transform.default.input] + } +} + +// This processor set resource attributes on spans +otelcol.processor.transform "default" { + error_mode = "ignore" + {{- if .CustomResourceAttributes }} + + trace_statements { + context = "resource" + statements = [ + {{- range $key, $value := .CustomResourceAttributes }} + `set(attributes["{{ $key }}"], "{{ $value }}")`, + {{- end }} + ] + } + {{- end }} + {{- if not .CustomResourceAttributes }} + // No CustomResourceAttributes set + {{- end }} + + output { + metrics = [otelcol.processor.batch.default.input] + logs = [otelcol.processor.batch.default.input] + traces = [otelcol.processor.batch.default.input] + } +} + +otelcol.processor.batch "default" { + output { + traces = [ + {{- range $i, $value := .Credentials }} + otelcol.exporter.otlphttp.default_{{$i}}.input, + {{- end}} + ] + } +} + +{{- range $i, $value := .Credentials }} + +otelcol.auth.basic "credentials_{{$i}}" { + username = {{ $value.User }} + password = {{ $value.Password }} +} + +otelcol.exporter.otlphttp "default_{{$i}}" { + client { + endpoint = {{ $value.Endpoint }} + auth = otelcol.auth.basic.credentials_{{$i}}.handler + } +} + +{{- end }} diff --git a/pkg/controllers/fixtures/additional_scrapping_config b/pkg/controllers/fixtures/additional_scrapping_config new file mode 100644 index 0000000..7aec347 --- /dev/null +++ b/pkg/controllers/fixtures/additional_scrapping_config @@ -0,0 +1,61 @@ +- job_name: "prometheus-scraper" + static_configs: + - targets: ["localhost:9090"] +- job_name: prometheusesNamespace/prometheus-podmonitorNamespace-monitoring-target/0 + honor_labels: true + kubernetes_sd_configs: + - role: service + namespaces: + names: + - prometheusesNamespace + scrape_interval: 30s + metrics_path: /federate + params: + match[]: + - '{federate="true", namespace="podmonitorNamespace"}' + relabel_configs: + - action: keep + source_labels: + - __meta_kubernetes_service_label_release + regex: ^.*-(ingress|cluster)-metrics$ + - source_labels: + - __meta_kubernetes_endpoint_address_target_kind + - __meta_kubernetes_endpoint_address_target_name + separator: ; + regex: Node;(.*) + replacement: ${1} + target_label: node + - source_labels: + - __meta_kubernetes_endpoint_address_target_kind + - __meta_kubernetes_endpoint_address_target_name + separator: ; + regex: Pod;(.*) + replacement: ${1} + target_label: pod + - source_labels: + - __meta_kubernetes_namespace + target_label: namespace + - source_labels: + - __meta_kubernetes_service_name + target_label: service + - source_labels: + - __meta_kubernetes_pod_name + target_label: pod + - source_labels: + - __meta_kubernetes_service_name + target_label: job + replacement: ${1} + - regex: pod + action: labeldrop + - regex: node + action: labeldrop + - regex: namespace + action: labeldrop + metric_relabel_configs: + - regex: federate + action: labeldrop + - regex: __replica__ + action: labeldrop + - regex: ^prometheus$ + action: labeldrop + enable_http2: false diff --git a/pkg/controllers/interfaces.go b/pkg/controllers/interfaces.go new file mode 100644 index 0000000..46baec8 --- /dev/null +++ b/pkg/controllers/interfaces.go @@ -0,0 +1,8 @@ +package controllers + +import "github.com/adevinta/observability-operator/pkg/grafanacloud" + +type GrafanaCloudClient interface { + GetStack(tenant string) (*grafanacloud.Stack, error) + GetTracesConnection(stack string) (int, string, error) +} diff --git a/pkg/controllers/interfaces_test.go b/pkg/controllers/interfaces_test.go new file mode 100644 index 0000000..7adb673 --- /dev/null +++ b/pkg/controllers/interfaces_test.go @@ -0,0 +1,16 @@ +package controllers + +import "github.com/adevinta/observability-operator/pkg/grafanacloud" + +type mockGrafanaCloudClient struct { + GetStackFunc func(string) (*grafanacloud.Stack, error) + GetTracesConnectionFunc func(string) (int, string, error) +} + +func (m *mockGrafanaCloudClient) GetStack(tenant string) (*grafanacloud.Stack, error) { + return m.GetStackFunc(tenant) +} + +func (m *mockGrafanaCloudClient) GetTracesConnection(stack string) (int, string, error) { + return m.GetTracesConnectionFunc(stack) +} diff --git a/pkg/controllers/metrics.go b/pkg/controllers/metrics.go new file mode 100644 index 0000000..d266ceb --- /dev/null +++ b/pkg/controllers/metrics.go @@ -0,0 +1,28 @@ +package controllers + +import ( + "github.com/prometheus/client_golang/prometheus" + "sigs.k8s.io/controller-runtime/pkg/metrics" +) + +var ( + podMonitorsErrors = prometheus.NewCounterVec( + prometheus.CounterOpts{ + Name: "podmonitor_errors", + Help: "Number of errors reconciling podmonitor objects", + }, []string{"name", "namespace"}, + ) + + prometheusErrors = prometheus.NewCounterVec( + prometheus.CounterOpts{ + Name: "prometheus_errors", + Help: "Number of errors reconciling prometheus objects/ingresses/svcs", + }, []string{"name", "namespace", "kind"}, + ) +) + +func init() { + prometheusErrors.WithLabelValues("default", "default", "default").Add(0) + podMonitorsErrors.WithLabelValues("default", "default").Add(0) + metrics.Registry.MustRegister(prometheusErrors, podMonitorsErrors) +} diff --git a/pkg/controllers/namespace_controller.go b/pkg/controllers/namespace_controller.go new file mode 100644 index 0000000..f8eeb37 --- /dev/null +++ b/pkg/controllers/namespace_controller.go @@ -0,0 +1,236 @@ +package controllers + +import ( + "context" + "fmt" + "strings" + "time" + + "github.com/go-logr/logr" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/labels" + types "k8s.io/apimachinery/pkg/types" + "k8s.io/client-go/util/workqueue" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/event" + "sigs.k8s.io/controller-runtime/pkg/handler" + "sigs.k8s.io/controller-runtime/pkg/reconcile" +) + +type GrafanaCloudUpdater interface { + InjectFluentdLokiConfiguration(context.Context) error +} + +type NamespaceReconciler struct { + client.Client + Log logr.Logger + + ExcludeWorkloadLabelSelector labels.Selector + ExcludeNamespaceLabelSelector labels.Selector + IgnoreApps []string + IgnoreNamespaces []string + TracesNamespace string + GrafanaCloudUpdater GrafanaCloudUpdater + GrafanaCloudClient GrafanaCloudClient + GrafanaCloudTracesToken string + ClusterName string + EnableVPA bool +} + +func (r *NamespaceReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { + log := r.Log.WithValues("namespace", req.Name) + if r.isIgnoredNamespace(req.Name) { + log.Info("Namespace is in the ignore list") + return ctrl.Result{}, nil + } + ns := corev1.Namespace{} + if err := r.Get(ctx, req.NamespacedName, &ns); err != nil { + log.Error(err, fmt.Sprintf("Namespace not found for %s ", req.Name)) + return ctrl.Result{}, client.IgnoreNotFound(err) + } + + if isExcludedByLabelSelector(r.ExcludeNamespaceLabelSelector, ns.Labels) { + return ctrl.Result{}, nil + } + if value, ok := ns.Labels[Config.logsLabelKey]; !ok || value != "disabled" { + err := r.GrafanaCloudUpdater.InjectFluentdLokiConfiguration(ctx) + if err != nil { + return ctrl.Result{Requeue: true, RequeueAfter: 30 * time.Second}, err + } + } + + if ns.Annotations[Config.tracesAnnotationKey] == "enabled" { + if _, err := reconcileTracesCollector(ctx, r.Client, log, r.GrafanaCloudClient, r.GrafanaCloudTracesToken, r.ClusterName, r.TracesNamespace, req.Name, r.IgnoreApps, r.ExcludeWorkloadLabelSelector, r.EnableVPA); err != nil { + log.Error(err, "Unable to reconcile traces collector") + return ctrl.Result{}, err + } + } else { + if _, err := r.deleteTracesCollector(ctx, req); err != nil { + log.Error(err, "Unable to delete traces collector") + return ctrl.Result{}, err + } + } + + if value, ok := ns.Labels[Config.metricsLabelKey]; !ok || value != "disabled" { + log.Info("Reconciling pod monitors") + return r.reconcilePodMonitors(ctx, req) + } + return ctrl.Result{}, nil +} + +func (r *NamespaceReconciler) reconcilePodMonitors(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { + log := r.Log.WithValues("namespace", req.Name) + ns := corev1.Namespace{} + if err := r.Get(ctx, req.NamespacedName, &ns); err != nil { + log.Info("Namespace object not found. Skipping...") + return ctrl.Result{}, client.IgnoreNotFound(err) + } + + podMonitors := &monitoringv1.PodMonitorList{} + if err := r.Client.List(ctx, podMonitors, client.InNamespace(req.Name)); err != nil { + log.Error(err, "Unable to list pod monitors") + return ctrl.Result{}, err + } + log.Info("listing podmonitors in the namespace") + for _, podMonitor := range podMonitors.Items { + err := r.reconcileNamespace(ctx, log, ns, podMonitor) + if err != nil { + log.Error(err, "Unable to update PodMonitor object", "PodMonitor", podMonitor.Name, "Namespace", podMonitor.Namespace) + return ctrl.Result{}, err + } else { + log.WithValues("podMonitorNamespace", podMonitor.Namespace, "podMonitorName", podMonitor.Name).Info("reconciled pod monitor") + } + } + + return ctrl.Result{}, nil +} + +func (r *NamespaceReconciler) isIgnoredNamespace(namespace string) bool { + for _, toFind := range r.IgnoreNamespaces { + if namespace == toFind { + return true + } + } + return false +} + +func (r *NamespaceReconciler) reconcileNamespace(ctx context.Context, log logr.Logger, namespace corev1.Namespace, podMonitor *monitoringv1.PodMonitor) error { + log = log.WithValues("namespace", podMonitor.Namespace, "name", podMonitor.Name) + result, err := ctrl.CreateOrUpdate( + ctx, + r.Client, + podMonitor, + func() error { + return updatePodMonitorAnnotation(ctx, r.Client, podMonitor) + }, + ) + if err != nil { + podMonitorsErrors.WithLabelValues(podMonitor.Name, podMonitor.Namespace).Inc() + log.Error(err, "failed to create or update pod monitor") + return err + } + log.WithValues("result", result).Info("created or updated pod monitor") + + return nil +} + +func updatePodMonitorAnnotation(ctx context.Context, cl client.Client, podMonitor *monitoringv1.PodMonitor) error { + namespace := &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: podMonitor.Namespace, + }, + } + err := cl.Get(ctx, types.NamespacedName{Name: podMonitor.Namespace}, namespace) + if err != nil { + return err + } + // We should just override this known annotation and keep the rest as it is + annotations := []string{Config.storageAnnotationKey, Config.alertManagerAnnotationKey, Config.remoteWriteAnnotationKey, Config.stackNameAnnotationKey} + for _, annotation := range annotations { + element, ok := namespace.Annotations[annotation] + if ok { + podMonitor.ObjectMeta.Annotations[annotation] = element + } + + } + return nil +} + +func (r *NamespaceReconciler) SetupWithManager(mgr ctrl.Manager, grafanaStackChangeEvents chan event.GenericEvent) error { + grafanaSource := &grafanaStackChangesSource{ + Client: r.Client, + changes: grafanaStackChangeEvents, + log: ctrl.Log.WithName("grafanaStackChangeWatchMap"), + } + return ctrl.NewControllerManagedBy(mgr). + For(&corev1.Namespace{}). + WatchesRawSource( + grafanaSource, + ). + Watches( + &corev1.Pod{}, + handler.EnqueueRequestsFromMapFunc( + func(ctx context.Context, a client.Object) []reconcile.Request { + return []reconcile.Request{{ + NamespacedName: types.NamespacedName{ + Name: a.GetNamespace(), + }, + }} + }, + ), + ). + Complete(r) +} + +type grafanaStackChangesSource struct { + Client client.Client + changes chan event.GenericEvent + log logr.Logger +} + +func (m *grafanaStackChangesSource) Start(ctx context.Context, wq workqueue.TypedRateLimitingInterface[reconcile.Request]) error { + go func() { + for e := range m.changes { + for _, req := range m.Map(e.Object) { + wq.Add(req) + } + } + }() + return nil +} + +// Map checks if the "stack changed event" matches one of the stacks +// configured in any of our namespaces, and enqueues a namespace +// reconciliation for the affected ones. +func (m *grafanaStackChangesSource) Map(event client.Object) []reconcile.Request { + var requests []reconcile.Request + + namespaces := corev1.NamespaceList{} + if err := m.Client.List(context.Background(), &namespaces); err != nil { + m.log.Error(err, "failed to list namespaces") + return requests + } + + changedStack := event.GetName() + for _, ns := range namespaces.Items { + stacks, ok := ns.GetAnnotations()[Config.stackNameAnnotationKey] + if !ok { + m.log.Info("namespace does not have stack annotation", "namespace", ns.Name) + continue + } + + for _, configuredStack := range strings.Split(stacks, ",") { + if configuredStack == changedStack { + requests = append(requests, reconcile.Request{ + NamespacedName: types.NamespacedName{ + Name: ns.Name, + }, + }) + } + } + } + return requests +} diff --git a/pkg/controllers/namespace_controller_test.go b/pkg/controllers/namespace_controller_test.go new file mode 100644 index 0000000..75a37f8 --- /dev/null +++ b/pkg/controllers/namespace_controller_test.go @@ -0,0 +1,575 @@ +package controllers + +import ( + "context" + "errors" + "os" + "testing" + + "github.com/adevinta/observability-operator/pkg/grafanacloud" + "github.com/adevinta/observability-operator/pkg/test_helpers" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + apilabels "k8s.io/apimachinery/pkg/labels" + "k8s.io/apimachinery/pkg/runtime" + types "k8s.io/apimachinery/pkg/types" + validatingfield "k8s.io/apimachinery/pkg/util/validation/field" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/event" + "sigs.k8s.io/controller-runtime/pkg/reconcile" +) + +type GrafanaCloudUpdaterFunc struct { + InjectFluentdLokiConfigurationCalls int + InjectFluentdLokiConfigurationFunc func(context.Context) error +} + +func (u *GrafanaCloudUpdaterFunc) InjectFluentdLokiConfiguration(ctx context.Context) error { + u.InjectFluentdLokiConfigurationCalls++ + if u.InjectFluentdLokiConfigurationFunc != nil { + return u.InjectFluentdLokiConfigurationFunc(ctx) + } + return errors.New("not implemented") +} + +func newDefaultNamespaceReconciler(t *testing.T, initialObjects ...runtime.Object) *NamespaceReconciler { + scheme := runtime.NewScheme() + _ = monitoringv1.AddToScheme(scheme) + + fakeClient := test_helpers.NewFakeClient(t, initialObjects...) + + os.Setenv("GRAFANA_CLOUD_TOKEN", "zzzz") + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + return nil, nil + }, + GetTracesConnectionFunc: func(stack string) (int, string, error) { + return 0, "", nil + }, + } + + return &NamespaceReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("Namespace"), + Client: fakeClient, + TracesNamespace: "observability", + ClusterName: "clustername", + GrafanaCloudClient: &grafanaCloudClient, + GrafanaCloudTracesToken: "GCO_TRACES_TOKEN", + GrafanaCloudUpdater: &GrafanaCloudUpdaterFunc{ + InjectFluentdLokiConfigurationFunc: func(context.Context) error { + return nil + }, + }, + } +} + +func TestNamespaceReconciliationFailsWhenGrafanaCloudConfigurationFails(t *testing.T) { + ns := newNamespace("my-namespace") + reconciler := newDefaultNamespaceReconciler(t, ns) + testError := errors.New("namespace reconciliation should fail when grafana cloud configuration fails") + gcUpdater := &GrafanaCloudUpdaterFunc{ + InjectFluentdLokiConfigurationFunc: func(context.Context) error { + return testError + }, + } + reconciler.GrafanaCloudUpdater = gcUpdater + result, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: "my-namespace"}}) + assert.Equal(t, testError, err) + assert.True(t, result.Requeue) +} + +func TestNamespaceReconciliationSucceedsWhenGrafanaCloudConfigurationSucceeds(t *testing.T) { + reconciler := newDefaultNamespaceReconciler(t) + gcUpdater := &GrafanaCloudUpdaterFunc{ + InjectFluentdLokiConfigurationFunc: func(context.Context) error { + return nil + }, + } + reconciler.GrafanaCloudUpdater = gcUpdater + result, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: "my-namespace"}}) + assert.Nil(t, err) + assert.False(t, result.Requeue) +} + +func TestShouldDoNothingWhenNamespaceIsIgnored(t *testing.T) { + reconciler := newDefaultNamespaceReconciler(t) + reconciler.IgnoreNamespaces = []string{"my-ignored-namespace"} + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "my-ignored-namespace"}}) + assert.Nil(t, err) + assert.Equal(t, ctrl.Result{}, result) +} + +func TestShouldHonorExcludeNamespaceSelectors(t *testing.T) { + pm1 := createPodMonitor("ignore-me", "test-pm") + annotations1 := map[string]string{Config.stackNameAnnotationKey: "wont-be-changed"} + pm1.Annotations = annotations1 + + pm2 := createPodMonitor("dont-ignore-me", "test-pm") + annotations2 := map[string]string{Config.stackNameAnnotationKey: "will-be-changed"} + pm2.Annotations = annotations2 + + pm3 := createPodMonitor("empty-selector-dont-ignore-me", "test-pm") + annotations3 := map[string]string{Config.stackNameAnnotationKey: "will-be-changed"} + pm3.Annotations = annotations3 + + ns1 := newNamespace("ignore-me") + ns1.Labels = map[string]string{"label": "ignoreme"} + ns1.Annotations = map[string]string{Config.stackNameAnnotationKey: "gcstack"} + ns2 := newNamespace("dont-ignore-me") + ns2.Labels = map[string]string{"label": "ignoreyou"} + ns2.Annotations = map[string]string{Config.stackNameAnnotationKey: "gcstack"} + ns3 := newNamespace("empty-selector-dont-ignore-me") + ns3.Labels = map[string]string{"label": "ignoreyou"} + ns3.Annotations = map[string]string{Config.stackNameAnnotationKey: "gcstack"} + + reconciler := newDefaultNamespaceReconciler(t, ns1, ns2, ns3, pm1, pm2, pm3) + + path := validatingfield.NewPath("metadata", "labels") + filter, err := apilabels.Parse("label=ignoreme", validatingfield.WithPath(path)) + assert.NoError(t, err) + reconciler.ExcludeNamespaceLabelSelector = filter + + t.Run(" Should ignore the namespace when the selector matches", func(t *testing.T) { + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "ignore-me"}}) + assert.NoError(t, err) + assert.Equal(t, ctrl.Result{}, result) + pm := &monitoringv1.PodMonitor{} + require.NoError(t, reconciler.Client.Get(context.Background(), types.NamespacedName{Name: "test-pm", Namespace: "ignore-me"}, pm)) + }) + t.Run(" Should reconcile the namespace when the selector doesn't match, and update the podMonitor annotation", func(t *testing.T) { + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "dont-ignore-me"}}) + assert.NoError(t, err) + assert.Equal(t, ctrl.Result{}, result) + pm := &monitoringv1.PodMonitor{} + require.NoError(t, reconciler.Client.Get(context.Background(), types.NamespacedName{Name: "test-pm", Namespace: "dont-ignore-me"}, pm)) + require.Equal(t, "gcstack", pm.Annotations[Config.stackNameAnnotationKey]) + }) + + t.Run(" Should ignore empty Selectors and reconcile the namespace, and update the podMonitor annotation", func(t *testing.T) { + // empty selector + filter, err := apilabels.Parse("", validatingfield.WithPath(path)) + assert.NoError(t, err) + reconciler.ExcludeNamespaceLabelSelector = filter + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "empty-selector-dont-ignore-me"}}) + assert.NoError(t, err) + assert.Equal(t, ctrl.Result{}, result) + pm := &monitoringv1.PodMonitor{} + require.NoError(t, reconciler.Client.Get(context.Background(), types.NamespacedName{Name: "test-pm", Namespace: "empty-selector-dont-ignore-me"}, pm)) + require.Equal(t, "gcstack", pm.Annotations[Config.stackNameAnnotationKey]) + }) + +} + +func TestShouldDoNothingWhenLabelsAreIgnored(t *testing.T) { + reconciler := newDefaultNamespaceReconciler(t) + reconciler.IgnoreNamespaces = []string{"my-ignored-namespace"} + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "my-ignored-namespace"}}) + assert.NoError(t, err) + assert.Equal(t, ctrl.Result{}, result) +} + +func TestShouldDoNotCreatePrometheusWhenMetricsAnnotationIsDisabled(t *testing.T) { + ns := newNamespace("my-namespace") + + ns.Annotations = map[string]string{Config.storageAnnotationKey: "grafanacloud"} + ns.Labels = map[string]string{Config.metricsLabelKey: "disabled"} + + reconciler := newDefaultNamespaceReconciler(t, ns, createSecretStub("platform-services"), createPodStub("test-pod", "my-namespace")) + gcUpdater := &GrafanaCloudUpdaterFunc{ + InjectFluentdLokiConfigurationFunc: func(context.Context) error { + return nil + }, + } + reconciler.GrafanaCloudUpdater = gcUpdater + + result, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: "my-namespace"}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 0) + assert.Equal(t, 1, gcUpdater.InjectFluentdLokiConfigurationCalls) + assert.Equal(t, ctrl.Result{}, result) +} + +func TestShouldDoNotInjectLogsAnnotationIsDisabled(t *testing.T) { + reconciler := newDefaultNamespaceReconciler( + t, + &corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "my-namespace", + Labels: map[string]string{ + Config.logsLabelKey: "disabled", + }, + }, + }) + gcUpdater := &GrafanaCloudUpdaterFunc{ + InjectFluentdLokiConfigurationFunc: func(context.Context) error { + return nil + }, + } + reconciler.GrafanaCloudUpdater = gcUpdater + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "my-namespace"}}) + assert.Nil(t, err) + assert.Equal(t, 0, gcUpdater.InjectFluentdLokiConfigurationCalls) +} + +func TestShouldDoNothingWhenNamespaceIsNotFound(t *testing.T) { + reconciler := newDefaultNamespaceReconciler( + t, + &corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "my-namespace", + }, + }) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "not-found"}}) + assert.Nil(t, err) + assert.Equal(t, ctrl.Result{}, result) +} + +func TestNamespaceReconciliationShouldNotUpdateOtherNamespacesPodMonitors(t *testing.T) { + reconciler := newDefaultNamespaceReconciler( + t, + &corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "my-namespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "new-storage", + }, + }, + }, + &monitoringv1.PodMonitor{ + ObjectMeta: metav1.ObjectMeta{ + Name: "some-pod-monitor", + Namespace: "other-namespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "some-storage", + }, + }, + }, + &monitoringv1.PodMonitor{ + ObjectMeta: metav1.ObjectMeta{ + Name: "some-pod-monitor", + Namespace: "my-namespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "some-storage", + }, + }, + }, + ) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "my-namespace"}}) + assert.Nil(t, err) + assert.Equal(t, ctrl.Result{}, result) + podMonitor := &monitoringv1.PodMonitor{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: "my-namespace", + Name: "some-pod-monitor", + }, + } + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(podMonitor), podMonitor)) + assert.Equal(t, "new-storage", podMonitor.Annotations[Config.storageAnnotationKey]) + + podMonitor = &monitoringv1.PodMonitor{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: "other-namespace", + Name: "some-pod-monitor", + }, + } + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(podMonitor), podMonitor)) + assert.Equal(t, "some-storage", podMonitor.Annotations[Config.storageAnnotationKey]) +} + +func TestShouldUpdatePodMonitorAnnotationWhenNamespaceChanges(t *testing.T) { + type TestCase struct { + Description string + Namespace corev1.Namespace + GivenPodMonitorAnnotations map[string]string + ExpectedPodMonitorAnnotations map[string]string + } + + testCases := []TestCase{ + { + Description: "When the podmonitor and the namespace have no storage annotation but others", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{ + "should-not-add-this": "irrelevant", + }, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-1", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-1", + }, + }, + { + Description: "When the pod monitor has more annotations and the namespace has no storage, but additional, annotations", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{ + "should-not-add-this": "irrelevant", + }, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-2", + Config.storageAnnotationKey: "storage", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-2", + Config.storageAnnotationKey: "storage", + }, + }, + { + Description: "When the pod monitor has more annotations and the neither the namespace nor the pod monitor have no storage annotation", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{}, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-3", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-3", + }, + }, + { + Description: "When the pod monitor has more annotations and the namespace has no storage annotation", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{}, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-4", + Config.storageAnnotationKey: "storage", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-4", + Config.storageAnnotationKey: "storage", + }, + }, + { + Description: "When the pod monitor has more annotations and the namespace has a storage annotation", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{ + Config.storageAnnotationKey: "federation,grafanacloud", + }, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.storageAnnotationKey: "old-value", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.storageAnnotationKey: "federation,grafanacloud", + }, + }, + { + Description: "When the namespace defines an empty storage", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{ + Config.storageAnnotationKey: "", + }, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.storageAnnotationKey: "old-value", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.storageAnnotationKey: "", + }, + }, + { + Description: "When adding the annotation for alertmanager", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{ + Config.storageAnnotationKey: "new-storage", + Config.alertManagerAnnotationKey: "test", + }, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.storageAnnotationKey: "new-storage", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.storageAnnotationKey: "new-storage", + Config.alertManagerAnnotationKey: "test", + }, + }, + { + Description: "When adding the annotation for grafanacloud", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Annotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintastacktest", + }, + }, + }, + GivenPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + }, + ExpectedPodMonitorAnnotations: map[string]string{ + "should-not-touch-this-annotation": "my-value-5", + Config.stackNameAnnotationKey: "adevintastacktest", + }, + }, + } + + for _, item := range testCases { + t.Run(item.Description, func(t *testing.T) { + item.Namespace.Name = "my-namespace" + podmonitor := createPodMonitor("my-namespace", "podmonitor") + podmonitor.ObjectMeta.Annotations = item.GivenPodMonitorAnnotations + + reconciler := newDefaultNamespaceReconciler(t, podmonitor, &item.Namespace) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "my-namespace"}}) + assert.NoError(t, err) + assert.Equal(t, ctrl.Result{}, result) + podMonitorList := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitorList)) + require.Len(t, podMonitorList.Items, 1) + assert.Equal(t, item.ExpectedPodMonitorAnnotations, podMonitorList.Items[0].ObjectMeta.Annotations) + }) + } +} + +func createConfigMapRulesStub(name, namespace string) *corev1.ConfigMap { + relabelConfigs := ` + - regex: ^prometheus$ + action: labeldrop + - regex: ^node_exporter$ + action: labeldrop + ` + + return &corev1.ConfigMap{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: namespace, + Labels: map[string]string{"app": name}, + }, + Data: map[string]string{ + "relabel-configs": relabelConfigs, + }, + } +} + +func createConfigMapBrokenRulesStub(name, namespace string) *corev1.ConfigMap { + relabelConfigs := ` + - regex: ^prometheus$ + action:labeldrop + - regex: ^node_exporter$ + action labeldrop + ` + + return &corev1.ConfigMap{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: namespace, + Labels: map[string]string{"app": name}, + }, + Data: map[string]string{ + "relabel-configs": relabelConfigs, + }, + } +} + +func TestGrafanaStackChangesTriggerNamespaceReconciler(t *testing.T) { + genEvent := func(s string) *event.GenericEvent { + return &event.GenericEvent{ + Object: &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: s, + }, + }, + } + } + genNamespace := func(name, stackAnnotation string) *corev1.Namespace { + return &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Annotations: map[string]string{ + Config.stackNameAnnotationKey: stackAnnotation, + }, + }, + } + } + + stackName := "mystack" + namespaceName := stackName + "-dev" + t.Run("when there are no namespaces, does not trigger reconcile", func(t *testing.T) { + client := test_helpers.NewFakeClient(t) + source := grafanaStackChangesSource{ + Client: client, + } + event := genEvent(stackName) + + reconcileEvents := source.Map(event.Object) + require.Len(t, reconcileEvents, 0) + }) + t.Run("when the namespace is annotated for one stack and it doesn't match, does not trigger reconcile", func(t *testing.T) { + client := test_helpers.NewFakeClient(t, genNamespace(namespaceName, stackName+"1")) + source := grafanaStackChangesSource{ + Client: client, + } + event := genEvent(stackName) + + reconcileEvents := source.Map(event.Object) + require.Len(t, reconcileEvents, 0) + }) + t.Run("when the namespace is annotated for one stack and it matches, triggers reconcile", func(t *testing.T) { + client := test_helpers.NewFakeClient(t, genNamespace(namespaceName, stackName)) + source := grafanaStackChangesSource{ + Client: client, + } + event := genEvent(stackName) + + reconcileEvents := source.Map(event.Object) + require.Len(t, reconcileEvents, 1) + assert.Equal(t, namespaceName, reconcileEvents[0].Name) + }) + t.Run("when the namespace is annotated for multiple stacks and none matches, does not trigger reconcile", func(t *testing.T) { + client := test_helpers.NewFakeClient(t, genNamespace(namespaceName, stackName+"2,"+stackName+"1")) + source := grafanaStackChangesSource{ + Client: client, + } + event := genEvent(stackName) + + reconcileEvents := source.Map(event.Object) + require.Len(t, reconcileEvents, 0) + }) + t.Run("when the namespace is annotated for multiple stacks and one matches, triggers reconcile", func(t *testing.T) { + client := test_helpers.NewFakeClient(t, genNamespace(namespaceName, stackName+","+stackName+"1")) + source := grafanaStackChangesSource{ + Client: client, + } + event := genEvent(stackName) + + reconcileEvents := source.Map(event.Object) + require.Len(t, reconcileEvents, 1) + assert.Equal(t, namespaceName, reconcileEvents[0].Name) + }) +} diff --git a/pkg/controllers/namespace_controller_traces_utils.go b/pkg/controllers/namespace_controller_traces_utils.go new file mode 100644 index 0000000..05efd4e --- /dev/null +++ b/pkg/controllers/namespace_controller_traces_utils.go @@ -0,0 +1,171 @@ +package controllers + +import ( + "context" + "fmt" + "strconv" + "time" + + "github.com/go-logr/logr" + apierrors "k8s.io/apimachinery/pkg/api/errors" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + + "k8s.io/apimachinery/pkg/labels" +) + +const ( + httpPort = int32(12345) + otelGrpcPort = int32(4317) + otelHttpPort = int32(4318) +) + +func reconcileTracesCollector(ctx context.Context, client client.Client, log logr.Logger, grafanaCloudClient GrafanaCloudClient, grafanaCloudTracesToken, clusterName, tracesNamespace, tenantNamespace string, ignoreApps []string, excludeLabelsSelector labels.Selector, enableVPA bool) (ctrl.Result, error) { + creds, err := getTracesCredentials(client, grafanaCloudClient, log, tenantNamespace) + if err != nil { + return ctrl.Result{}, err + } + + collector, err := NewTracesCollector( + tenantNamespace, + tracesNamespace, + clusterName, + enableVPA, + WithHTTPPort(httpPort), + WithServicePorts( + servicePort{Name: "otel-http", Port: otelHttpPort}, + servicePort{Name: "otel-grpc", Port: otelGrpcPort}, + ), + AlloyWithOTelCredentials(creds...), + ) + if err != nil { + return ctrl.Result{}, err + } + + var filters []podFilter + for _, ignoredApp := range ignoreApps { + filters = append(filters, excludePodsOnLabel("app", ignoredApp)) + } + + if excludeLabelsSelector != nil && !excludeLabelsSelector.Empty() { + filters = append(filters, filterBySelector(excludeLabelsSelector)) + } + + pods, err := checkNamespaceHasActionableWorkloads(client, log, tenantNamespace, filters...) + if err != nil { + return ctrl.Result{}, err + } + if len(pods) == 0 { + // No valid workload, remove the collector. If it does + // not exist this won't fail and will be a noop. + for _, obj := range collector.ObjectsToDelete() { + log := log.WithValues("object", fmt.Sprintf("%T", obj), "name", obj.GetName(), "namespace", obj.GetNamespace()) + + err := client.Delete(ctx, obj) + if err != nil { + if !apierrors.IsNotFound(err) { + log.Error(err, "Failed to delete object in namespace") + return ctrl.Result{Requeue: true, RequeueAfter: 30 * time.Second}, err + } + // The object is not found we ignore it + } else { + log.Info("Object %T deleted in namespace %s", obj, obj.GetNamespace()) + } + } + + return ctrl.Result{}, nil + } + + if err := collector.CreateOrUpdateSecret(ctx, client, grafanaCloudTracesToken); err != nil { + log.Error(err, "Error when reconciling secret") + return ctrl.Result{}, err + } + + if err := collector.CreateOrUpdateConfigMap(ctx, client); err != nil { + log.Error(err, "Error when reconciling configmap") + return ctrl.Result{}, err + } + + if err := collector.CreateOrUpdateService(ctx, client); err != nil { + log.Error(err, "Error when reconciling service") + return ctrl.Result{}, err + } + + if err := collector.CreateOrUpdateDeployment(ctx, client); err != nil { + log.Error(err, "Error when reconciling deployment") + return ctrl.Result{}, err + } + + if err := collector.CreateOrUpdateNetworkPolicy(ctx, client); err != nil { + log.Error(err, "Error when reconciling network policy") + return ctrl.Result{}, err + } + + if enableVPA { + if err := collector.CreateOrUpdateVPA(ctx, client); err != nil { + log.Error(err, "Error when reconciling VPA") + return ctrl.Result{}, err + } + } + + log.Info("All traces objects successfully reconciled") + return ctrl.Result{}, nil +} + +func (r *NamespaceReconciler) deleteTracesCollector(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { + tc, _ := NewTracesCollector( + req.Name, // tenantNamespace + r.TracesNamespace, + r.ClusterName, + r.EnableVPA, + WithHTTPPort(httpPort), + WithServicePorts( + servicePort{Name: "otel-http", Port: otelHttpPort}, + servicePort{Name: "otel-grpc", Port: otelGrpcPort}, + ), + ) + objs := tc.ObjectsToDelete() + + for _, obj := range objs { + log := r.Log.WithValues("object", fmt.Sprintf("%T", obj), "name", obj.GetName(), "namespace", obj.GetNamespace()) + + err := r.Delete(ctx, obj) + if err != nil { + if !apierrors.IsNotFound(err) { + log.Error(err, "Failed to delete object in namespace") + return ctrl.Result{Requeue: true, RequeueAfter: 30 * time.Second}, err + } + // The object is not found we ignore it + } else { + log.Info("Object deleted in namespace") + } + } + return ctrl.Result{}, nil +} + +func getTracesCredentials(client client.Client, grafanaCloudClient GrafanaCloudClient, log logr.Logger, namespace string) ([]OTelCredentials, error) { + grafanaStackNames, err := lookupGrafanaStacks(client, namespace) + if err != nil || len(grafanaStackNames) < 1 { + log.Error(err, "Failed to lookup grafana stacks") + return []OTelCredentials{}, err + } + + var out []OTelCredentials + for _, stack := range grafanaStackNames { + otlpId, otlpUrl, err := grafanaCloudClient.GetTracesConnection(stack) + if err != nil { + log.Error(err, "Failed to get Grafana stack connections info for stack", "grafana_stacks", stack) + continue + } + + out = append(out, OTelCredentials{ + User: strconv.Itoa(otlpId), + Endpoint: fmt.Sprintf("\"%s/otlp\"", otlpUrl), + }) + } + if len(out) == 0 { + err := fmt.Errorf("no grafana stack credentials retrieved") + return []OTelCredentials{}, err + } + return out, nil +} diff --git a/pkg/controllers/namespace_controller_traces_utils_test.go b/pkg/controllers/namespace_controller_traces_utils_test.go new file mode 100644 index 0000000..2f74a85 --- /dev/null +++ b/pkg/controllers/namespace_controller_traces_utils_test.go @@ -0,0 +1,575 @@ +package controllers + +import ( + "context" + "fmt" + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + appv1 "k8s.io/api/apps/v1" + corev1 "k8s.io/api/core/v1" + networkingv1 "k8s.io/api/networking/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + apilabels "k8s.io/apimachinery/pkg/labels" + types "k8s.io/apimachinery/pkg/types" + validatingfield "k8s.io/apimachinery/pkg/util/validation/field" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/reconcile" +) + +func createDefaultTracesSecret(tracesNamespace, objectName string) *corev1.Secret { + return &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: objectName, + Namespace: tracesNamespace, + Labels: map[string]string{ + "app.kubernetes.io/name": objectName, + "app.kubernetes.io/managed-by": "grafana-cloud-operator", + "app.kubernetes.io/version": "v1.2.1", + }, + }, + Data: map[string][]byte{ + "grafana-cloud-traces-token": []byte("GCO_TRACES_TOKEN"), + }, + Type: corev1.SecretTypeOpaque, + } +} + +func createDefaultTenantPod(namespace string) *corev1.Pod { + return &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: namespace, + }, + } +} + +func createIgnoredTenantPod(namespace, appLabel string) *corev1.Pod { + return &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "ignore-me-pod", + Namespace: namespace, + Labels: map[string]string{ + "app": appLabel, + }, + }, + } +} + +func TestNamespaceControllerCreatesAlloyTracesCollector(t *testing.T) { + tests := []struct { + name string + annotations map[string]string + expectedObjects int + }{ + { + name: "Alloy traces collector objects, are created, when there are workloads in the tenant namespace", + annotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + }, + }, + } + + tracesNs := newNamespace("observability") + ns := newNamespace("teamtest-dev") + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + ns.Annotations = tt.annotations + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs, createDefaultTenantPod(ns.Name), createIgnoredTenantPod(ns.Namespace, "ignored-app")) + path := validatingfield.NewPath("metadata", "labels") + filter, err := apilabels.Parse("app=ignored-app", validatingfield.WithPath(path)) + require.NoError(t, err) + reconciler.ExcludeWorkloadLabelSelector = filter + reconciler.EnableVPA = true + + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + + secrets := &corev1.SecretList{} + err = reconciler.Client.List(context.Background(), secrets, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, secrets.Items, 1) + assert.Len(t, secrets.Items[0].Data, 1) + assert.Contains(t, secrets.Items[0].Data, "grafana-cloud-traces-token") + + services := &corev1.ServiceList{} + err = reconciler.Client.List(context.Background(), services, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, services.Items, 1) + assert.Len(t, services.Items[0].Spec.Ports, 3) + + configMaps := &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, configMaps.Items, 1) + assert.Len(t, configMaps.Items[0].Data, 1) + assert.Contains(t, configMaps.Items[0].Data, "config.alloy") + + deployments := &appv1.DeploymentList{} + err = reconciler.Client.List(context.Background(), deployments, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, deployments.Items, 1) + + networkPolicies := &networkingv1.NetworkPolicyList{} + err = reconciler.Client.List(context.Background(), networkPolicies, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, networkPolicies.Items, 1) + assert.Len(t, networkPolicies.Items[0].Spec.Ingress, 1) + assert.Len(t, networkPolicies.Items[0].Spec.Ingress[0].From, 1) + assert.Len(t, networkPolicies.Items[0].Spec.Ingress[0].From[0].NamespaceSelector.MatchLabels, 1) + assert.Contains(t, networkPolicies.Items[0].Spec.Ingress[0].From[0].NamespaceSelector.MatchLabels, "kubernetes.io/metadata.name") + + vpas := &vpav1.VerticalPodAutoscalerList{} + err = reconciler.Client.List(context.Background(), vpas, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, vpas.Items, 1) + assert.Equal(t, vpas.Items[0].Spec.TargetRef.Kind, "Deployment") + assert.Equal(t, vpas.Items[0].Spec.TargetRef.Name, deployments.Items[0].Name) + assert.Equal(t, *vpas.Items[0].Spec.UpdatePolicy.UpdateMode, vpav1.UpdateMode("Auto")) + }) + } +} + +func TestNamespaceControllerDoesNotCreateAlloyTracesCollector(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("teamtest-dev") + + ignoreAppName := "ignored-app" + + tests := []struct { + name string + tenantNsAnnotations map[string]string + pod *corev1.Pod + }{ + { + name: "Alloy traces collector objects, are not created, when traces annotation value is different than enabled", + tenantNsAnnotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "disabled", + }, + }, + { + name: "Alloy traces collector objects, are not created, when the traces annotation is missing", + tenantNsAnnotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + }, + }, + { + name: "Alloy traces collector objects, are not created, when there are no workloads", + tenantNsAnnotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + }, + }, + { + name: "Alloy traces collector objects, are not created, when the only workloads are ignored", + tenantNsAnnotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + }, + pod: createIgnoredTenantPod(ns.Name, ignoreAppName), + }, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + ns.Annotations = tt.tenantNsAnnotations + + var reconciler *NamespaceReconciler + if tt.pod != nil { + reconciler = newDefaultNamespaceReconciler(t, ns, tracesNs, tt.pod) + } else { + reconciler = newDefaultNamespaceReconciler(t, ns, tracesNs) + } + path := validatingfield.NewPath("metadata", "labels") + filter, err := apilabels.Parse(fmt.Sprintf("app=%s", ignoreAppName), validatingfield.WithPath(path)) + require.NoError(t, err) + reconciler.ExcludeWorkloadLabelSelector = filter + + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + + secrets := &corev1.SecretList{} + err = reconciler.Client.List(context.Background(), secrets, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + assert.Len(t, secrets.Items, 0) + + services := &corev1.ServiceList{} + err = reconciler.Client.List(context.Background(), services, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + assert.Len(t, services.Items, 0) + + configMaps := &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + assert.Len(t, configMaps.Items, 0) + + deployments := &appv1.DeploymentList{} + err = reconciler.Client.List(context.Background(), deployments, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, deployments.Items, 0) + + networkPolicies := &networkingv1.NetworkPolicyList{} + err = reconciler.Client.List(context.Background(), networkPolicies, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + assert.Len(t, networkPolicies.Items, 0) + + vpas := &vpav1.VerticalPodAutoscalerList{} + err = reconciler.Client.List(context.Background(), vpas, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, vpas.Items, 0) + }) + } +} + +func TestNamespaceControllerAlloyConfiguration(t *testing.T) { + type mockGrafanaResponses struct { + StackID int + URL string + Error error + } + tests := []struct { + name string + annotations map[string]string + labels map[string]string + objectLabels map[string]string + mockgc mockGrafanaResponses + expectedConfigMapData []string + }{ + { + name: "Alloy traces collector configmap, and secret, have the correct data for the Alloy config. The configmap must have a Grafana stack related to the annotation", + annotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintadummy", + Config.tracesAnnotationKey: "enabled", + }, + labels: map[string]string{ + "app.kubernetes.io/name": "alloy-team-test-dev", + "app.kubernetes.io/managed-by": "grafana-cloud-operator", + "app.kubernetes.io/version": "v1.2.1", + }, + objectLabels: map[string]string{ + "app.kubernetes.io/name": "alloy-team-test-dev", + "app.kubernetes.io/managed-by": "grafana-cloud-operator", + "app.kubernetes.io/version": "v1.2.1", + }, + mockgc: mockGrafanaResponses{ + URL: "https://test-otlp-endpoint-adevintadummy.com", + }, + expectedConfigMapData: []string{"https://test-otlp-endpoint-adevintadummy.com/otlp"}, + }, + { + name: "Alloy traces collector configmap, and secret, must have the correct data for the Alloy config. The configmap should be configured with a Grafana stack despite not having an annotation", + annotations: map[string]string{ + Config.tracesAnnotationKey: "enabled", + Config.stackNameAnnotationKey: "adevintatest", + }, + objectLabels: map[string]string{ + "app.kubernetes.io/name": "alloy-team-test-dev", + "app.kubernetes.io/managed-by": "grafana-cloud-operator", + "app.kubernetes.io/version": "v1.2.1", + }, + mockgc: mockGrafanaResponses{ + URL: "https://test-otlp-endpoint-adevintateamtest.com", + }, + expectedConfigMapData: []string{"https://test-otlp-endpoint-adevintateamtest.com/otlp"}, + }, + { + name: "Alloy traces collector configmap, must have custom resource attributes processor with k8s.cluster.name and k8s.namespace.name", + annotations: map[string]string{ + Config.tracesAnnotationKey: "enabled", + Config.stackNameAnnotationKey: "adevintatest", + }, + objectLabels: map[string]string{ + "app.kubernetes.io/name": "alloy-team-test-dev", + "app.kubernetes.io/managed-by": "grafana-cloud-operator", + "app.kubernetes.io/version": "v1.2.1", + }, + mockgc: mockGrafanaResponses{ + URL: "https://test-otlp-endpoint-adevintadummy.com", + }, + expectedConfigMapData: []string{ + "otelcol.processor.transform", + "trace_statements", + "set(attributes[\"k8s.cluster.name\"], \"clustername\"", + "set(attributes[\"k8s.namespace.name\"], \"team-test-dev\"", + }, + }, + } + + tracesNs := newNamespace("observability") + ns := newNamespace("team-test-dev") + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + + ns.Labels = tt.labels + ns.Annotations = tt.annotations + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs, createDefaultTenantPod(ns.Name)) + grafanaCloudClient := mockGrafanaCloudClient{ + GetTracesConnectionFunc: func(stack string) (int, string, error) { + return tt.mockgc.StackID, tt.mockgc.URL, tt.mockgc.Error + }, + } + + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedObjectName := "alloy-" + ns.Name + expectedTracesNamespace := reconciler.TracesNamespace + expectedSecret := createDefaultTracesSecret(expectedTracesNamespace, expectedObjectName) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + + secrets := &corev1.SecretList{} + err = reconciler.Client.List(context.Background(), secrets, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + require.Len(t, secrets.Items, 1) + expectedSecret.ResourceVersion = secrets.Items[0].ResourceVersion + assert.Contains(t, string(secrets.Items[0].Data["grafana-cloud-traces-token"]), "GCO_TRACES_TOKEN") + + assert.Equal(t, *expectedSecret, secrets.Items[0]) + + configMaps := &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + for _, expectedData := range tt.expectedConfigMapData { + assert.Contains(t, configMaps.Items[0].Data["config.alloy"], expectedData) + } + }) + } +} + +func TestNamespaceCollectorUpdatesObjectMeta(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + ns.Annotations = map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + } + + grafanaCloudClient := mockGrafanaCloudClient{ + GetTracesConnectionFunc: func(stack string) (int, string, error) { + return 0, "https://test-otlp-endpoint-" + stack + ".com", nil + }, + } + + t.Run("Reconcile updates object when there are changes", func(t *testing.T) { + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs, createDefaultTenantPod(ns.Name)) + reconciler.EnableVPA = true + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedTracesNamespace := reconciler.TracesNamespace + + _, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + + secrets := &corev1.SecretList{} + err = reconciler.Client.List(context.Background(), secrets, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + assert.Len(t, secrets.Items, 1) + + services := &corev1.ServiceList{} + err = reconciler.Client.List(context.Background(), services, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + assert.Len(t, services.Items, 1) + + configMaps := &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + require.Len(t, configMaps.Items, 1) + require.Contains(t, configMaps.Items[0].Data, "config.alloy") + + deployments := &appv1.DeploymentList{} + err = reconciler.Client.List(context.Background(), deployments, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, deployments.Items, 1) + + networkPolicies := &networkingv1.NetworkPolicyList{} + err = reconciler.Client.List(context.Background(), networkPolicies, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + assert.Len(t, networkPolicies.Items, 1) + + vpas := &vpav1.VerticalPodAutoscalerList{} + err = reconciler.Client.List(context.Background(), vpas, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, vpas.Items, 1) + assert.Equal(t, vpas.Items[0].Spec.TargetRef.Kind, "Deployment") + assert.Equal(t, vpas.Items[0].Spec.TargetRef.Name, deployments.Items[0].Name) + assert.Equal(t, *vpas.Items[0].Spec.UpdatePolicy.UpdateMode, vpav1.UpdateMode("Auto")) + + originalConfig := configMaps.Items[0].Data["config.alloy"] + + existingNs := &corev1.Namespace{} + err = reconciler.Client.Get(context.Background(), types.NamespacedName{Name: ns.Name}, existingNs) + require.NoError(t, err) + existingNs.Annotations[Config.stackNameAnnotationKey] = "adevintaotherstack" + err = reconciler.Client.Update(context.Background(), existingNs) + require.NoError(t, err) + + _, err = reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + + secrets = &corev1.SecretList{} + err = reconciler.Client.List(context.Background(), secrets, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + assert.Len(t, secrets.Items, 1) + + services = &corev1.ServiceList{} + err = reconciler.Client.List(context.Background(), services, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + assert.Len(t, services.Items, 1) + + configMaps = &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + require.Contains(t, configMaps.Items[0].Data, "config.alloy") + assert.NotEqual(t, originalConfig, configMaps.Items[0].Data["config.alloy"]) + + deployments = &appv1.DeploymentList{} + err = reconciler.Client.List(context.Background(), deployments, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, deployments.Items, 1) + + networkPolicies = &networkingv1.NetworkPolicyList{} + err = reconciler.Client.List(context.Background(), networkPolicies, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + assert.Len(t, networkPolicies.Items, 1) + + vpas = &vpav1.VerticalPodAutoscalerList{} + err = reconciler.Client.List(context.Background(), vpas, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, vpas.Items, 1) + assert.Equal(t, vpas.Items[0].Spec.TargetRef.Kind, "Deployment") + assert.Equal(t, vpas.Items[0].Spec.TargetRef.Name, deployments.Items[0].Name) + assert.Equal(t, *vpas.Items[0].Spec.UpdatePolicy.UpdateMode, vpav1.UpdateMode("Auto")) + }) +} + +func TestTracesCollectorReconcileDisableTenantIfTracesAnnotationChange(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + ns.Annotations = map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + } + + verifyReconcile := func(reconciler *NamespaceReconciler, namespace string, length int) { + secrets := &corev1.SecretList{} + err := reconciler.Client.List(context.Background(), secrets, client.InNamespace(namespace)) + require.NoError(t, err) + assert.Len(t, secrets.Items, length) + + services := &corev1.ServiceList{} + err = reconciler.Client.List(context.Background(), services, client.InNamespace(namespace)) + require.NoError(t, err) + assert.Len(t, services.Items, length) + + configMaps := &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(namespace)) + require.NoError(t, err) + assert.Len(t, configMaps.Items, length) + + deployments := &appv1.DeploymentList{} + err = reconciler.Client.List(context.Background(), deployments, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, deployments.Items, length) + + networkPolicies := &networkingv1.NetworkPolicyList{} + err = reconciler.Client.List(context.Background(), networkPolicies, client.InNamespace(namespace)) + require.NoError(t, err) + assert.Len(t, networkPolicies.Items, length) + + vpas := &vpav1.VerticalPodAutoscalerList{} + err = reconciler.Client.List(context.Background(), vpas, client.InNamespace(reconciler.TracesNamespace)) + require.NoError(t, err) + require.Len(t, vpas.Items, length) + } + + t.Run("Reconcile must create the traces resources when annotation is enabled and delete it when it is different from enabled", func(t *testing.T) { + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs, createDefaultTenantPod(ns.Name)) + reconciler.EnableVPA = true + + expectedTracesNamespace := reconciler.TracesNamespace + + _, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + verifyReconcile(reconciler, expectedTracesNamespace, 1) + + existingNs := &corev1.Namespace{} + err = reconciler.Client.Get(context.Background(), types.NamespacedName{Name: ns.Name}, existingNs) + require.NoError(t, err) + existingNs.Annotations[Config.tracesAnnotationKey] = "whatevervalue" + + err = reconciler.Client.Update(context.Background(), existingNs) + require.NoError(t, err) + + _, err = reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + verifyReconcile(reconciler, expectedTracesNamespace, 0) + }) + + t.Run("Reconcile must create the traces resources when annotation is enabled and delete it when it is not set", func(t *testing.T) { + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs, createDefaultTenantPod(ns.Name)) + reconciler.EnableVPA = true + + expectedTracesNamespace := reconciler.TracesNamespace + + _, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + verifyReconcile(reconciler, expectedTracesNamespace, 1) + + existingNs := &corev1.Namespace{} + err = reconciler.Client.Get(context.Background(), types.NamespacedName{Name: ns.Name}, existingNs) + require.NoError(t, err) + delete(existingNs.Annotations, Config.tracesAnnotationKey) + + err = reconciler.Client.Update(context.Background(), existingNs) + require.NoError(t, err) + + _, err = reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + verifyReconcile(reconciler, expectedTracesNamespace, 0) + }) +} + +func TestTracesCollectorDeploymentIsAnnotatedForObservability(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + ns.Annotations = map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + } + + t.Run("Alloy Pod is annotated to be able to gather metrics from it", func(t *testing.T) { + expectedAnnotations := map[string]string{ + "prometheus.io/path": "/metrics", + "prometheus.io/port": "12345", + "prometheus.io/scrape": "true", + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs, createDefaultTenantPod(ns.Name)) + + expectedTracesNamespace := reconciler.TracesNamespace + + _, err := reconciler.Reconcile(context.Background(), reconcile.Request{NamespacedName: types.NamespacedName{Name: ns.Name}}) + require.NoError(t, err) + + deployments := &appv1.DeploymentList{} + err = reconciler.Client.List(context.Background(), deployments, client.InNamespace(expectedTracesNamespace)) + require.NoError(t, err) + + require.Len(t, deployments.Items, 1) + assert.Equal(t, expectedAnnotations, deployments.Items[0].Spec.Template.Annotations) + }) +} diff --git a/pkg/controllers/pod_controller.go b/pkg/controllers/pod_controller.go new file mode 100644 index 0000000..7d69dd0 --- /dev/null +++ b/pkg/controllers/pod_controller.go @@ -0,0 +1,387 @@ +package controllers + +import ( + "context" + "fmt" + "slices" + "strconv" + "strings" + + "github.com/go-logr/logr" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + corev1 "k8s.io/api/core/v1" + apierrors "k8s.io/apimachinery/pkg/api/errors" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" + "k8s.io/apimachinery/pkg/runtime" + types "k8s.io/apimachinery/pkg/types" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/client/apiutil" + "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil" + + "k8s.io/apimachinery/pkg/labels" +) + +var ( + standardRelabeling = []*monitoringv1.RelabelConfig{ + { + Action: "labelmap", + Regex: "__meta_kubernetes_pod_label_(.+)", + Replacement: "${1}", + }, + { + // In the legacy prometheus-local-container deployment, the presence of prometheus.io/port label was overriding the port discovery done by default by prometheus. + // This was done by our configuration instead of prometheus itself. + // In most cases, not having this configuration is good enoguh. Prometheus will scrape, by default, all defined container ports: https://github.com/prometheus/prometheus/blob/04145344991acd7571c715bdbe1c2c6dfec9f871/discovery/kubernetes/pod.go#L242 + // We though, discovered a case where this is not working. When the port is not registered inside the pod definition. There are occurences of this setup in our clusters. + // In order to be backward compatible, we introduce the same configuration as in prometheus-local-container and honor the prometheus.io/port annotation contract. + // In future releases we may want to deprecate this behaviour + Action: "replace", + // Prometheus drops the replacement if the value does not match: + // https://github.com/prometheus/prometheus/blob/04145344991acd7571c715bdbe1c2c6dfec9f871/pkg/relabel/relabel.go#L215 + // so, if the label is absent, empty, or a string, the regexp will not match and the original __address__ will be left intact + Regex: `(.+):(?:\d+);(\d+)`, + Replacement: "${1}:${2}", + Separator: ";", + SourceLabels: []string{ + "__address__", + "__meta_kubernetes_pod_annotation_prometheus_io_port", + }, + TargetLabel: "__address__", + }, + } +) + +// PodReconciler reconciles a Pod object +type PodReconciler struct { + client.Client + Log logr.Logger + Scheme *runtime.Scheme + ExcludeWorkloadLabelSelector labels.Selector + ExcludeNamespaceLabelSelector labels.Selector + IgnoreApps []string + IgnoreNamespaces []string + PrometheusNamespace string + TracesNamespace string + GrafanaCloudClient GrafanaCloudClient + GrafanaCloudTracesToken string + ClusterName string + EnableVPA bool +} + +func getOwner(obj client.Object) client.Object { + for _, owner := range obj.GetOwnerReferences() { + if owner.Controller != nil && *owner.Controller { + u := &unstructured.Unstructured{} + u.SetKind(owner.Kind) + u.SetAPIVersion(owner.APIVersion) + u.SetName(owner.Name) + u.SetNamespace(obj.GetNamespace()) + return u + } + } + return obj +} + +func isExcludedByLabelSelector(selector labels.Selector, lbs map[string]string) bool { + if selector != nil && !selector.Empty() { + if selector.Matches(labels.Set(lbs)) { + return true + } + } + return false +} + +func (r *PodReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { + pod := corev1.Pod{} + if err := r.Get(ctx, req.NamespacedName, &pod); err != nil { + if apierrors.IsNotFound(err) { + ns := corev1.Namespace{} + if err := r.Get(ctx, types.NamespacedName{Name: req.Namespace}, &ns); err != nil { + r.Log.Error(err, fmt.Sprintf("Namespace %s not found for pod %s", req.Namespace, req.Name)) + return ctrl.Result{}, err + } + if ns.Annotations[Config.tracesAnnotationKey] == "enabled" { + if _, err := reconcileTracesCollector(ctx, r.Client, r.Log, r.GrafanaCloudClient, r.GrafanaCloudTracesToken, r.ClusterName, r.TracesNamespace, req.Namespace, r.IgnoreApps, r.ExcludeWorkloadLabelSelector, r.EnableVPA); err != nil { + r.Log.Error(err, "Unable to reconcile traces collector") + return ctrl.Result{}, err + } + } + return ctrl.Result{}, nil + } + return ctrl.Result{}, err + } + + ns := corev1.Namespace{} + if err := r.Get(ctx, types.NamespacedName{Name: req.Namespace}, &ns); err != nil { + r.Log.Error(err, fmt.Sprintf("Namespace %s not found for pod %s", req.Namespace, req.Name)) + return ctrl.Result{}, err + } + + if isExcludedByLabelSelector(r.ExcludeNamespaceLabelSelector, ns.Labels) { + r.Log.V(10).Info("Namespace is excluded by label selector") + return ctrl.Result{}, nil + } + + owner := getOwner(&pod) + if err := r.Get(ctx, client.ObjectKeyFromObject(owner), owner); err != nil { + return ctrl.Result{}, client.IgnoreNotFound(err) + } + + if owner.GetObjectKind().GroupVersionKind().Kind == "ReplicaSet" { + owner = getOwner(owner) + if err := r.Get(ctx, client.ObjectKeyFromObject(owner), owner); err != nil { + return ctrl.Result{}, client.IgnoreNotFound(err) + } + } + + if owner.GetObjectKind().GroupVersionKind().Kind == "Pod" { + // avoid creating one pod monitor per pod in the namespace. + // we will review the strategy for those pods in the future. + // After a quick analysis, we don't have any pod that would cause us any trouble here + r.Log.Info("pod has no owner, skipping", "namespace", pod.Namespace, "name", pod.Name) + return ctrl.Result{}, nil + } + + err := r.reconcilePodMonitor(owner) + if err != nil { + return ctrl.Result{}, err + } + + if ns.Annotations[Config.tracesAnnotationKey] == "enabled" { + if _, err := reconcileTracesCollector(ctx, r.Client, r.Log, r.GrafanaCloudClient, r.GrafanaCloudTracesToken, r.ClusterName, r.TracesNamespace, req.Namespace, r.IgnoreApps, r.ExcludeWorkloadLabelSelector, r.EnableVPA); err != nil { + r.Log.Error(err, "Unable to reconcile traces collector") + return ctrl.Result{}, err + } + } + return ctrl.Result{}, nil +} + +func (r *PodReconciler) reconcilePodMonitor(owner client.Object) error { + monitor := &monitoringv1.PodMonitor{ + ObjectMeta: metav1.ObjectMeta{ + Name: owner.GetName(), + Namespace: owner.GetNamespace(), + }, + } + gvk := owner.GetObjectKind().GroupVersionKind() + log := r.Log.WithValues("apiVersion", gvk.GroupVersion().String(), "kind", gvk.Kind, "namespace", owner.GetNamespace(), "name", owner.GetName()) + + if r.needsToBeScrapped(owner) { + r.createOrUpdatePodMonitor(log, owner, monitor) + return nil + } + + log.Info("does not need scrapping") + log = log.WithValues("needsToBeScraped", false) + if r.monitorExists(monitor) { + if r.monitorIsManaged(monitor, owner) { + if err := r.deleteMonitor(monitor); err != nil { + log.Error(err, "failed to delete no longer needed monitor") + return err + } + log.Info("deleting no longer needed monitor") + return nil + } + log.Info("Monitor is not managed by the controller, skipping") + return nil + } + return nil +} + +func (r *PodReconciler) createOrUpdatePodMonitor(log logr.Logger, owner client.Object, monitor *monitoringv1.PodMonitor) { + storage, err := getAvailableStorageFromNamespace(r.Client, monitor.Namespace, r.Log) + if err != nil { + r.Log.Info("Could not determinate the Storage from the Namespace %s, skipping PodMonitor sync", monitor.Namespace) + return + } + sampleLimit := getPodMonitorSampleLimit(r.Client, owner, r.Log) + ctx := context.Background() + result, err := ctrl.CreateOrUpdate( + ctx, + r.Client, + monitor, + func() error { + monitor.ObjectMeta.Labels = getLabels(owner, monitor) + monitor.ObjectMeta.Annotations = map[string]string{ + Config.storageAnnotationKey: storage, + Config.accountAnnotationKey: monitor.Namespace, + Config.podMonitorAnnotationKey: owner.GetNamespace() + "-" + owner.GetName(), + } + err := controllerutil.SetOwnerReference(owner, monitor, r.Scheme) + if err != nil { + log.Error(err, "failed to assign owner reference to deployment") + } + + u := &unstructured.Unstructured{} + switch o := owner.(type) { + case *unstructured.Unstructured: + u = o + default: + u.Object, err = runtime.DefaultUnstructuredConverter.ToUnstructured(owner) + if err != nil { + return err + } + } + unstructuredSelector, ok, err := unstructured.NestedMap(u.Object, "spec", "selector") + if err != nil { + return err + } + if !ok { + return fmt.Errorf("could not find spec.selector in pod monitor owner") + } + selector := metav1.LabelSelector{} + err = runtime.DefaultUnstructuredConverter.FromUnstructured(unstructuredSelector, &selector) + if err != nil { + return err + } + + monitor.Spec = monitoringv1.PodMonitorSpec{ + Selector: selector, + SampleLimit: uint64(sampleLimit), + PodMetricsEndpoints: []monitoringv1.PodMetricsEndpoint{ + { + Path: getPath(owner), + Params: getParams(owner), + HonorLabels: true, + RelabelConfigs: standardRelabeling, + }, + }, + NamespaceSelector: monitoringv1.NamespaceSelector{ + MatchNames: []string{owner.GetNamespace()}, + }, + } + return updatePodMonitorAnnotation(ctx, r.Client, monitor) + }, + ) + if err != nil { + log.Error(err, "failed to create or update pod monitor") + } else { + log.WithValues("result", result).Info("created pod monitor") + } +} +func (r *PodReconciler) monitorIsManaged(monitor *monitoringv1.PodMonitor, owner client.Object) bool { + ctx := context.Background() + m := monitoringv1.PodMonitor{} + _ = r.Get(ctx, types.NamespacedName{Namespace: monitor.Namespace, Name: monitor.Name}, &m) + gvk, err := apiutil.GVKForObject(owner, r.Scheme) + if err != nil { + // TODO handle the error (log? return an actual error?) + return false + } + for _, ownerReferenced := range m.GetOwnerReferences() { + if ownerReferenced.Kind == gvk.Kind && ownerReferenced.Name == owner.GetName() && ownerReferenced.UID == owner.GetUID() { + return true + } + } + return false +} +func (r *PodReconciler) deleteMonitor(monitor *monitoringv1.PodMonitor) error { + ctx := context.Background() + return r.Delete(ctx, monitor) +} + +func (r *PodReconciler) monitorExists(monitor *monitoringv1.PodMonitor) bool { + ctx := context.Background() + m := monitoringv1.PodMonitor{} + if err := r.Get(ctx, types.NamespacedName{Namespace: monitor.Namespace, Name: monitor.Name}, &m); err != nil { + return false + } + return true +} + +func (r *PodReconciler) needsToBeScrapped(object client.Object) bool { + if slices.Contains(r.IgnoreNamespaces, object.GetNamespace()) { + return false + } + + for _, ignoredApp := range r.IgnoreApps { + if object.GetLabels()["app"] == ignoredApp { + return false + } + } + + if isExcludedByLabelSelector(r.ExcludeWorkloadLabelSelector, object.GetLabels()) { + return false + } + + scrape, ok := getPodAnnotations(object)["prometheus.io/scrape"].(string) + return ok && scrape == "true" +} + +func getPath(object client.Object) string { + pathIntf, ok := getPodAnnotations(object)["prometheus.io/path"] + if ok { + if path, ok := pathIntf.(string); ok { + return path + } + } + return "/metrics" +} + +// getParams will extract parameters from prometheus.io/param_* annotations +func getParams(object client.Object) map[string][]string { + params := map[string][]string{} + for annotation, valueIntf := range getPodAnnotations(object) { + if strings.HasPrefix(annotation, "prometheus.io/param_") { + param := strings.TrimPrefix(annotation, "prometheus.io/param_") + if value, ok := valueIntf.(string); ok { + // k8s doesn't support duplicate keys, so only one value is supported + params[param] = []string{value} + } + } + } + return params +} + +func getPodAnnotations(object client.Object) map[string]interface{} { + u, err := runtime.DefaultUnstructuredConverter.ToUnstructured(object) + if err != nil { + return map[string]interface{}{} + } + path := []string{"spec", "template", "metadata", "annotations"} + switch o := object.(type) { + case *corev1.Pod: + path = []string{"metadata", "annotations"} + case *unstructured.Unstructured: + if o.GetKind() == "Pod" { + path = []string{"metadata", "annotations"} + } + } + annotations, _, _ := unstructured.NestedMap(u, path...) + if annotations == nil { + return map[string]interface{}{} + } + return annotations +} + +func getLabels(owner client.Object, monitor *monitoringv1.PodMonitor) map[string]string { + labels := map[string]string{ + Config.accountLabelKey: monitor.Namespace, + } + for k, v := range owner.GetLabels() { + labels[k] = v + } + return labels +} + +func getPodMonitorSampleLimit(k8sclient client.Client, object client.Object, log logr.Logger) int { + value, ok := getPodAnnotations(object)[Config.podSampleLimitAnnotation] + if !ok { + return 4500 + } + sampleLimitInt, err := strconv.Atoi(value.(string)) + if err != nil { + log.Error(err, "Could not convert value in "+Config.podSampleLimitAnnotation+" annotation to integer, setting it to default 4500") + return 4500 + } + return sampleLimitInt + +} + +func (r *PodReconciler) SetupWithManager(mgr ctrl.Manager) error { + return ctrl.NewControllerManagedBy(mgr). + For(&corev1.Pod{}). + Complete(r) +} diff --git a/pkg/controllers/pod_controller_test.go b/pkg/controllers/pod_controller_test.go new file mode 100644 index 0000000..01ac733 --- /dev/null +++ b/pkg/controllers/pod_controller_test.go @@ -0,0 +1,1135 @@ +package controllers + +import ( + "bytes" + "context" + "encoding/json" + "os" + "testing" + + "github.com/adevinta/observability-operator/pkg/grafanacloud" + "github.com/adevinta/observability-operator/pkg/test_helpers" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + "github.com/prometheus/prometheus/pkg/labels" + "github.com/prometheus/prometheus/pkg/relabel" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "gopkg.in/yaml.v3" + appv1 "k8s.io/api/apps/v1" + corev1 "k8s.io/api/core/v1" + networkingv1 "k8s.io/api/networking/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" + apilabels "k8s.io/apimachinery/pkg/labels" + "k8s.io/apimachinery/pkg/runtime" + "k8s.io/apimachinery/pkg/types" + validatingfield "k8s.io/apimachinery/pkg/util/validation/field" + "k8s.io/utils/ptr" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +func assertLabelValue(t *testing.T, name, value string, labelset labels.Labels) { + t.Helper() + for _, label := range labelset { + if label.Name == name { + assert.Equal(t, value, label.Value) + return + } + } + t.Errorf("did not find label with name %s", name) +} + +func TestRelabeling(t *testing.T) { + b, err := json.Marshal(standardRelabeling) + require.NoError(t, err) + // hack to translate from very similar prometheus-operator format to prometheus-config one + b = bytes.Replace(b, []byte(`"sourceLabels"`), []byte(`"source_labels"`), -1) + b = bytes.Replace(b, []byte(`"targetLabel"`), []byte(`"target_label"`), -1) + cfg := []*relabel.Config{} + require.NoError(t, yaml.NewDecoder(bytes.NewReader(b)).Decode(&cfg)) + + assertLabelValue(t, "__address__", "127.0.0.1:6400", relabel.Process(labels.Labels{{Name: "__address__", Value: "127.0.0.1:6400"}}, cfg...)) + assertLabelValue(t, "__address__", "127.0.0.1:2021", relabel.Process(labels.Labels{{Name: "__address__", Value: "127.0.0.1:6400"}, {Name: "__meta_kubernetes_pod_annotation_prometheus_io_port", Value: "2021"}}, cfg...)) + assertLabelValue(t, "__address__", "127.0.0.1:6400", relabel.Process(labels.Labels{{Name: "__address__", Value: "127.0.0.1:6400"}, {Name: "__meta_kubernetes_pod_annotation_prometheus_io_port", Value: "some-port"}}, cfg...)) + assertLabelValue(t, "__address__", "127.0.0.1:6400", relabel.Process(labels.Labels{{Name: "__address__", Value: "127.0.0.1:6400"}, {Name: "__meta_kubernetes_pod_annotation_prometheus_io_port", Value: ""}}, cfg...)) + assertLabelValue(t, "some_label", "value", relabel.Process(labels.Labels{{Name: "__meta_kubernetes_pod_label_some_label", Value: "value"}}, cfg...)) +} + +func TestPodReconcilerRetrievesOwner(t *testing.T) { + namespace := &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: "bar", + }, + } + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "ReplicaSet", + Name: "foo-v1", + Controller: ptr.To(true), + }, + }, + }, + } + + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + + deploy := &appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo", + Namespace: "bar", + }, + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + t.Run("When the pod is owned by a deployment", func(t *testing.T) { + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, replicaSet, deploy) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "foo-pod", Namespace: "bar"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 1) + assert.Equal(t, "foo", podMonitors.Items[0].Name) + assert.EqualValues( + t, + metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + podMonitors.Items[0].Spec.Selector, + ) + }) + t.Run("When the pod is owned by a statefulset", func(t *testing.T) { + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "StatefulSet", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + + statefulSet := &appv1.StatefulSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo", + Namespace: "bar", + }, + Spec: appv1.StatefulSetSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, statefulSet) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "foo-pod", Namespace: "bar"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 1) + assert.Equal(t, "foo", podMonitors.Items[0].Name) + assert.EqualValues( + t, + metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + podMonitors.Items[0].Spec.Selector, + ) + }) + t.Run("When the pod is owned by a non-existing statefulset", func(t *testing.T) { + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "StatefulSet", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + statefulSet := &appv1.StatefulSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "other", + Namespace: "bar", + }, + Spec: appv1.StatefulSetSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, statefulSet) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "foo-pod", Namespace: "bar"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 0) + }) + t.Run("When the pod is owned by a non-existing deployment", func(t *testing.T) { + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "ReplicaSet", + Name: "foo-v1", + Controller: ptr.To(true), + }, + }, + }, + } + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, replicaSet) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 0) + }) + + t.Run("When the pod is has no owner", func(t *testing.T) { + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + } + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, replicaSet) + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Name: "foo-pod", Namespace: "bar"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 0) + }) +} + +func TestDeploymentCreatesPodMonitorWithNamespaceStorageAnnotations(t *testing.T) { + namespace := &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: "bar", + }, + } + + t.Run("when the namespace is not annotated", func(t *testing.T) { + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "ReplicaSet", + Name: "foo-v1", + Controller: ptr.To(true), + }, + }, + }, + } + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + deploy := &appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo", + Namespace: "bar", + }, + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, replicaSet, deploy) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "bar", Name: "foo-pod"}}) + assert.Nil(t, err) + assert.NotNil(t, result) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 1) + assert.Equal(t, "grafanacloud", podMonitors.Items[0].Annotations[Config.storageAnnotationKey]) + }) + + t.Run("when the namespace is annotated", func(t *testing.T) { + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "ReplicaSet", + Name: "foo-v1", + Controller: ptr.To(true), + }, + }, + }, + } + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + deploy := &appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo", + Namespace: "bar", + }, + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + reconciler := newDefaultPodReconciler(t, createSecretStub("prometheusesNamespace"), namespace, pod, replicaSet, deploy) + namespace.Annotations = map[string]string{ + Config.storageAnnotationKey: "new-storage", + } + require.NoError(t, reconciler.Client.Update(context.Background(), namespace)) + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "bar", Name: "foo-pod"}}) + assert.Nil(t, err) + assert.NotNil(t, result) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 1) + assert.Equal(t, "new-storage", podMonitors.Items[0].Annotations[Config.storageAnnotationKey]) + }) +} + +func generatePodObjectsWithLabels(namespaceName string, prometheusScrape string, labels map[string]string) []runtime.Object { + var annotations map[string]string + + namespace := newNamespace(namespaceName) + + if prometheusScrape != "" { + annotations = map[string]string{ + "prometheus.io/scrape": prometheusScrape, + } + } + + deployment := &appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "deploymentName", + Namespace: namespace.Name, + Labels: labels, + }, + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: annotations, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: namespace.Name, + Labels: labels, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: deployment.Name, + Controller: ptr.To(true), + }, + }, + }, + } + + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: namespace.Name, + Annotations: annotations, + Labels: labels, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "ReplicaSet", + Name: replicaSet.Name, + Controller: ptr.To(true), + }, + }, + }, + } + + return []runtime.Object{namespace, deployment, replicaSet, pod} + +} + +func generatePodObjects(namespaceName string, prometheusScrape string) []runtime.Object { + return generatePodObjectsWithLabels(namespaceName, prometheusScrape, map[string]string{}) +} + +func TestPodMonitorReconcile(t *testing.T) { + t.Run("when there is a Pod, a PodMonitor is created", func(t *testing.T) { + objects := generatePodObjects("my-namespace", "true") + reconciler := newDefaultPodReconciler(t, objects...) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "my-namespace", Name: "foo-pod"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + err = reconciler.Client.List(context.Background(), podMonitors) + require.NoError(t, err) + assert.Len(t, podMonitors.Items, 1) + }) + t.Run("when the scrape annotation is false, there are no PodMonitors created", func(t *testing.T) { + objects := generatePodObjects("my-namespace", "false") + reconciler := newDefaultPodReconciler(t, objects...) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "my-namespace", Name: "foo-pod"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + err = reconciler.Client.List(context.Background(), podMonitors) + require.NoError(t, err) + assert.Len(t, podMonitors.Items, 0) + + }) + t.Run("when there is no scrape annotation, there are no PodMonitors created", func(t *testing.T) { + objects := generatePodObjects("my-namespace", "") + reconciler := newDefaultPodReconciler(t, objects...) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "my-namespace", Name: "foo-pod"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + err = reconciler.Client.List(context.Background(), podMonitors) + require.NoError(t, err) + assert.Len(t, podMonitors.Items, 0) + + }) + t.Run("when the namespace is ignored, there are no PodMonitors created", func(t *testing.T) { + objects := generatePodObjects("my-namespace", "true") + reconciler := newDefaultPodReconciler(t, objects...) + reconciler.IgnoreNamespaces = []string{"my-namespace"} + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "my-namespace", Name: "foo-pod"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + err = reconciler.Client.List(context.Background(), podMonitors) + require.NoError(t, err) + assert.Len(t, podMonitors.Items, 0) + }) + + t.Run("when the labels are ignored, there are no PodMonitors created", func(t *testing.T) { + objects := generatePodObjectsWithLabels("my-namespace", "true", map[string]string{"appLabel": "skip"}) + reconciler := newDefaultPodReconciler(t, objects...) + path := validatingfield.NewPath("metadata", "labels") + filter, err := apilabels.Parse("appLabel=skip", validatingfield.WithPath(path)) + require.NoError(t, err) + reconciler.ExcludeWorkloadLabelSelector = filter + + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "my-namespace", Name: "foo-pod"}}) + require.NoError(t, err) + + podMonitors := &monitoringv1.PodMonitorList{} + err = reconciler.Client.List(context.Background(), podMonitors) + require.NoError(t, err) + assert.Len(t, podMonitors.Items, 0) + }) + +} + +func TestDeployment(t *testing.T) { + deployment := &appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo", + Namespace: "bar", + UID: "123", + }, + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{}, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: deployment.Name, + + Controller: ptr.To(true), + }, + }, + }, + } + podMonitor := &monitoringv1.PodMonitor{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "foo", + Namespace: "bar", + OwnerReferences: []metav1.OwnerReference{{ + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + UID: "123", + }}, + Labels: map[string]string{ + Config.podMonitorLabelKey: "bar-foo", + Config.accountLabelKey: "bar", + }, + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.accountAnnotationKey: "bar", + }, + }, + Spec: monitoringv1.PodMonitorSpec{ + PodMetricsEndpoints: []monitoringv1.PodMetricsEndpoint{ + { + Path: "/metrics", + HonorLabels: true, + RelabelConfigs: standardRelabeling, + }, + }, + Selector: metav1.LabelSelector{ + MatchLabels: map[string]string{ + "random_label": "true", + }, + }, + NamespaceSelector: monitoringv1.NamespaceSelector{MatchNames: []string{"bar"}}, + SampleLimit: 9000, + }, + } + namespace := &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: "bar", + }, + } + t.Run("when the annotation for scrape is not properly set, the podmonitor instance is not created", func(t *testing.T) { + reconciler := newDefaultPodReconciler(t, pod, namespace) + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "bar", Name: "foo"}}) + require.NoError(t, err) + assert.NotNil(t, result) + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + assert.Len(t, podMonitors.Items, 0) + }) + + t.Run("once the annotation for scrape is set, the podmonitor instance is created", func(t *testing.T) { + reconciler := newDefaultPodReconciler(t, pod, namespace, podMonitor) + + deployment.Spec.Template.ObjectMeta.Annotations = map[string]string{ + "prometheus.io/scrape": "true", + Config.podSampleLimitAnnotation: "9000", + } + err := reconciler.Client.Create(context.Background(), deployment) + require.NoError(t, err) + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "bar", Name: "foo-pod"}}) + require.NoError(t, err) + assert.NotNil(t, result) + + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + require.Len(t, podMonitors.Items, 1) + assert.Equal(t, deployment.Name, podMonitors.Items[0].Name) + assert.Equal(t, deployment.Namespace, podMonitors.Items[0].Namespace) + assert.Equal(t, *deployment.Spec.Selector, podMonitors.Items[0].Spec.Selector) + assert.Len(t, podMonitors.Items[0].OwnerReferences, 1) + assert.Equal(t, podMonitors.Items[0].GetOwnerReferences()[0].APIVersion, "apps/v1") + assert.Equal(t, podMonitors.Items[0].GetOwnerReferences()[0].Kind, "Deployment") + assert.Equal(t, podMonitors.Items[0].GetOwnerReferences()[0].Name, "foo") + require.Len(t, podMonitors.Items[0].Spec.PodMetricsEndpoints, 1) + assert.Equal(t, "/metrics", podMonitors.Items[0].Spec.PodMetricsEndpoints[0].Path) + assert.True(t, podMonitors.Items[0].Spec.PodMetricsEndpoints[0].HonorLabels) + assert.Equal(t, podMonitors.Items[0].Spec.SampleLimit, uint64(9000)) + }) + + t.Run("that if the annotation is changed to not match, the podmonitor is removed", func(t *testing.T) { + reconciler := newDefaultPodReconciler(t, pod, namespace, podMonitor, deployment) + + deployment.Spec.Template.ObjectMeta.Annotations = map[string]string{ + "prometheus.io/scrape": "false", + } + require.NoError(t, reconciler.Client.Update(context.Background(), deployment)) + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "bar", Name: "foo-pod"}}) + assert.Nil(t, err) + assert.NotNil(t, result) + podMonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podMonitors)) + assert.Len(t, podMonitors.Items, 0) + }) +} + +func runGetPathTest(t *testing.T, expected string, object client.Object) { + t.Helper() + assert.Equal(t, expected, getPath(object)) +} + +func TestGetPath(t *testing.T) { + t.Run("with a Deployment and custom path", func(t *testing.T) { + runGetPathTest(t, "/mypath", &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/path": "/mypath", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment and other annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "other": "annotation", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment with no annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{}, + }, + }, + }) + }) + t.Run("with a DaemonSet and custom path", func(t *testing.T) { + runGetPathTest(t, "/mypath", &appv1.DaemonSet{ + Spec: appv1.DaemonSetSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/path": "/mypath", + }, + }, + }, + }, + }) + }) + t.Run("with a DaemonSet and other annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &appv1.DaemonSet{ + Spec: appv1.DaemonSetSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "other": "annotation", + }, + }, + }, + }, + }) + }) + t.Run("with a DaemonSet with no annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &appv1.DaemonSet{ + Spec: appv1.DaemonSetSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{}, + }, + }, + }) + }) + t.Run("with a Pod and custom path", func(t *testing.T) { + runGetPathTest(t, "/mypath", &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/path": "/mypath", + }, + }, + }) + }) + t.Run("with a Pod and other annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "other": "annotation", + }, + }, + }) + }) + t.Run("with a Pod with no annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{}, + }) + }) + + t.Run("with an unstructured StatefulSet and custom path", func(t *testing.T) { + runGetPathTest(t, "/mypath", &unstructured.Unstructured{ + Object: map[string]interface{}{ + "kind": "StatefulSet", + "spec": map[string]interface{}{ + "template": map[string]interface{}{ + "metadata": map[string]interface{}{ + "annotations": map[string]interface{}{ + "prometheus.io/path": "/mypath", + }, + }, + }, + }, + }, + }) + }) + t.Run("with an unstructured StatefulSet and other annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &unstructured.Unstructured{ + Object: map[string]interface{}{ + "kind": "StatefulSet", + "spec": map[string]interface{}{ + "template": map[string]interface{}{ + "metadata": map[string]interface{}{ + "annotations": map[string]interface{}{ + "other": "annotation", + }, + }, + }, + }, + }, + }) + }) + t.Run("with an unstructured StatefulSet with no annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &unstructured.Unstructured{ + Object: map[string]interface{}{ + "kind": "StatefulSet", + }, + }) + }) + + t.Run("with an unstructured Pod and custom path", func(t *testing.T) { + runGetPathTest(t, "/mypath", &unstructured.Unstructured{ + Object: map[string]interface{}{ + "kind": "Pod", + "metadata": map[string]interface{}{ + "annotations": map[string]interface{}{ + "prometheus.io/path": "/mypath", + }, + }, + }, + }) + }) + t.Run("with an unstructured Pod and other annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &unstructured.Unstructured{ + Object: map[string]interface{}{ + "kind": "Pod", + "metadata": map[string]interface{}{ + "annotations": map[string]interface{}{ + "other": "annotation", + }, + }, + }, + }) + }) + t.Run("with an unstructured Pod with no annotation", func(t *testing.T) { + runGetPathTest(t, "/metrics", &unstructured.Unstructured{ + Object: map[string]interface{}{ + "kind": "Pod", + }, + }) + }) +} + +func runGetParamsTest(t *testing.T, expected map[string][]string, object client.Object) { + t.Helper() + assert.Equal(t, expected, getParams(object)) +} + +func TestGetParams(t *testing.T) { + // only testing deployments here, as other objects' parsing is already tested in TestGetPath + t.Run("with a Deployment and custom parameter", func(t *testing.T) { + runGetParamsTest(t, map[string][]string{"format": {"prometheus"}}, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/param_format": "prometheus", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment and two custom parameters", func(t *testing.T) { + runGetParamsTest(t, map[string][]string{"format": {"prometheus"}, "foo": {"bar"}}, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/param_format": "prometheus", + "prometheus.io/param_foo": "bar", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment and other annotation", func(t *testing.T) { + runGetParamsTest(t, map[string][]string{}, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "other": "annotation", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment with no annotation", func(t *testing.T) { + runGetParamsTest(t, map[string][]string{}, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{}, + }, + }, + }) + }) +} + +func TestGetLabels(t *testing.T) { + deployment := appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "deployName", + Namespace: "deployNamespace", + Labels: map[string]string{ + "foo": "bar", + }, + }, + } + podmonitor := &monitoringv1.PodMonitor{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "monitorName", + Namespace: "monitorNamespace", + }, + } + assert.EqualValues(t, map[string]string{ + Config.accountLabelKey: "monitorNamespace", + "foo": "bar", + }, getLabels(&deployment, podmonitor)) +} + +func runGetPodMonitorsSampleLimitTest(t *testing.T, expected int, object client.Object) { + reconciler := newDefaultTestPodMonitorReconciler(t) + t.Helper() + val := getPodMonitorSampleLimit(reconciler.Client, object, reconciler.Log) + assert.Equal(t, expected, val) +} + +func TestGetPodMonitorSampleLimit(t *testing.T) { + // only testing deployments here, as other objects' parsing is already tested in TestGetPath + t.Run("with a Deployment and custom parameter", func(t *testing.T) { + runGetPodMonitorsSampleLimitTest(t, 9000, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + Config.podSampleLimitAnnotation: "9000", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment and empty annotation", func(t *testing.T) { + runGetPodMonitorsSampleLimitTest(t, 4500, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + Config.podSampleLimitAnnotation: "", + }, + }, + }, + }, + }) + }) + t.Run("with a Deployment with no annotation", func(t *testing.T) { + runGetPodMonitorsSampleLimitTest(t, 4500, &appv1.Deployment{ + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{}, + }, + }, + }) + }) + t.Run("with a Pod with the annotation", func(t *testing.T) { + runGetPodMonitorsSampleLimitTest(t, 9000, &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + Config.podSampleLimitAnnotation: "9000", + }, + }, + }) + }) +} + +func newDefaultPodReconciler(t *testing.T, initialObjects ...runtime.Object) *PodReconciler { + scheme := runtime.NewScheme() + _ = monitoringv1.AddToScheme(scheme) + _ = appv1.AddToScheme(scheme) + + fakeClient := test_helpers.NewFakeClient(t, initialObjects...) + os.Setenv("GRAFANA_CLOUD_TOKEN", "zzzz") + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + return nil, nil + }, + GetTracesConnectionFunc: func(stack string) (int, string, error) { + return 0, "", nil + }, + } + + return &PodReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("Namespace"), + Client: fakeClient, + Scheme: scheme, + ClusterName: "clustername", + TracesNamespace: "observability", + GrafanaCloudClient: &grafanaCloudClient, + GrafanaCloudTracesToken: "GCO_TRACES_TOKEN", + } +} + +func TestReconcileTracesCollector(t *testing.T) { + tracesNs := newNamespace("observability") + tenantNs := newNamespace("teamtest-dev") + + tenantNs.Annotations = map[string]string{ + Config.stackNameAnnotationKey: "adevintaobs", + Config.tracesAnnotationKey: "enabled", + } + + pod := &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-pod", + Namespace: tenantNs.Name, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "ReplicaSet", + Name: "foo-v1", + Controller: ptr.To(true), + }, + }, + }, + } + replicaSet := &appv1.ReplicaSet{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo-v1", + Namespace: tenantNs.Name, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "apps/v1", + Kind: "Deployment", + Name: "foo", + Controller: ptr.To(true), + }, + }, + }, + } + deploy := &appv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: "foo", + Namespace: tenantNs.Name, + }, + Spec: appv1.DeploymentSpec{ + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + "prometheus.io/scrape": "true", + }, + }, + }, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{"random_label": "true"}, + }, + }, + } + + verifyReconcile := func(reconciler *PodReconciler, namespace string, length int) { + secrets := &corev1.SecretList{} + err := reconciler.Client.List(context.Background(), secrets, client.InNamespace(namespace)) + require.NoError(t, err) + require.Len(t, secrets.Items, length) + + services := &corev1.ServiceList{} + err = reconciler.Client.List(context.Background(), services, client.InNamespace(namespace)) + require.NoError(t, err) + require.Len(t, services.Items, length) + + configMaps := &corev1.ConfigMapList{} + err = reconciler.Client.List(context.Background(), configMaps, client.InNamespace(namespace)) + require.NoError(t, err) + require.Len(t, configMaps.Items, length) + + networkPolicies := &networkingv1.NetworkPolicyList{} + err = reconciler.Client.List(context.Background(), networkPolicies, client.InNamespace(namespace)) + require.NoError(t, err) + require.Len(t, networkPolicies.Items, length) + } + + t.Run("there is a collector when a pod is present in the tenant namespace", func(t *testing.T) { + reconciler := newDefaultPodReconciler(t, tenantNs, tracesNs, pod, replicaSet, deploy) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: tenantNs.Name, Name: "foo-pod"}}) + require.NoError(t, err) + + verifyReconcile(reconciler, tracesNs.Name, 1) + }) + t.Run("there is no collector when no pod is present anymore in the tenant namespace", func(t *testing.T) { + reconciler := newDefaultPodReconciler(t, tenantNs, tracesNs, pod, replicaSet, deploy) + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: tenantNs.Name, Name: "foo-pod"}}) + require.NoError(t, err) + + verifyReconcile(reconciler, tracesNs.Name, 1) + + err = reconciler.Client.Delete(context.Background(), pod) + require.NoError(t, err) + err = reconciler.Client.Delete(context.Background(), replicaSet) + require.NoError(t, err) + err = reconciler.Client.Delete(context.Background(), deploy) + require.NoError(t, err) + + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: tenantNs.Name, Name: "foo-pod"}}) + require.NoError(t, err) + + verifyReconcile(reconciler, tracesNs.Name, 0) + }) +} diff --git a/pkg/controllers/podmonitor_controller.go b/pkg/controllers/podmonitor_controller.go new file mode 100644 index 0000000..7506c2b --- /dev/null +++ b/pkg/controllers/podmonitor_controller.go @@ -0,0 +1,956 @@ +package controllers + +import ( + "context" + "encoding/json" + "errors" + "fmt" + "os" + "regexp" + "strings" + "time" + + "github.com/adevinta/observability-operator/pkg/grafanacloud" + "github.com/go-logr/logr" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + promcommonconfig "github.com/prometheus/common/config" + promconfig "github.com/prometheus/prometheus/config" + "github.com/prometheus/prometheus/discovery/kubernetes" + "gopkg.in/yaml.v3" + autoscaling "k8s.io/api/autoscaling/v1" + corev1 "k8s.io/api/core/v1" + k8sErrors "k8s.io/apimachinery/pkg/api/errors" + "k8s.io/apimachinery/pkg/api/resource" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/runtime" + "k8s.io/apimachinery/pkg/types" + "k8s.io/apimachinery/pkg/util/intstr" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil" +) + +// PrometheusInstances is the structure definition to set DD metrics +type PrometheusInstances struct { + PrometheusURL string `json:"prometheus_url"` + Namespace string `json:"namespace"` + Tags []string `json:"tags"` + Metrics []string `json:"metrics"` +} + +type PrometheusAdditionalScrapeConfig []*promconfig.ScrapeConfig + +var defaultResourceRequirements corev1.ResourceRequirements = corev1.ResourceRequirements{ + Requests: corev1.ResourceList{ + corev1.ResourceCPU: resource.MustParse("100m"), + corev1.ResourceMemory: resource.MustParse("128Mi"), + }, + Limits: corev1.ResourceList{ + corev1.ResourceCPU: resource.MustParse("200m"), + corev1.ResourceMemory: resource.MustParse("256Mi"), + }, +} + +// DockerImage holds docker image registry reference +type DockerImage struct { + Name string + Tag string +} + +// PodMonitorReconciler reconciles a PodMonitor object +type PodMonitorReconciler struct { + client.Client + Log logr.Logger + Scheme *runtime.Scheme + ClusterName string + Region string + PrometheusNamespace string + PrometheusExposedDomain string + GrafanaCloudCredentials string + NodeSelectorTarget string + PrometheusDockerImage DockerImage + GrafanaCloudClient GrafanaCloudClient + EnableMetricsRemoteWrite bool + EnableVpa bool + PrometheusServiceAccountName string + PrometheusPodPriorityClassName string + PrometheusMonitoringTarget string + PrometheusExtraExternalLabels map[string]string +} + +func (r *PodMonitorReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) { + monitor := monitoringv1.PodMonitor{} + + if err := r.Get(ctx, req.NamespacedName, &monitor); err != nil { + return ctrl.Result{}, client.IgnoreNotFound(err) + } + log := r.Log.WithValues("podmonitor", monitor.Name, "namespace", monitor.Namespace) + stop, err := r.handleFinalizers(&monitor) + if stop { + if err == nil { + return ctrl.Result{}, nil + } + // If there was an error retry after 5 minutes again + return ctrl.Result{Requeue: true, RequeueAfter: 5}, err + } + err = r.reconcilePodMonitor(log, monitor) + if err != nil { + return ctrl.Result{Requeue: true, RequeueAfter: 5}, err + } + + return ctrl.Result{}, err +} + +func isPodMonitorBeingDeleted(monitor monitoringv1.PodMonitor) bool { + return !monitor.ObjectMeta.DeletionTimestamp.IsZero() +} + +func (r *PodMonitorReconciler) deletePrometheusIfNotRequired(monitor monitoringv1.PodMonitor) error { + ctx := context.Background() + monitors := &monitoringv1.PodMonitorList{} + // TODO: Theoretically we can use client.MatchingFields() as listOpts to match annotations, so we would + // not need the for loop afterwards + if err := r.List(ctx, monitors, client.InNamespace(monitor.Namespace)); err != nil { + if err != nil { + return err + } + } + for _, m := range monitors.Items { + if isGrafanaCloudStorageEnabled(m.ObjectMeta) && !isPodMonitorBeingDeleted(*m) { + return nil + } + } + + // There is no monitor to scrape, so we need to remove the prometheus + prometheus := newPrometheusObjectDef(monitor.Namespace, r.PrometheusNamespace) + return r.Delete(ctx, prometheus) +} + +func findInAnnotation(slice []string, sliceItem string) bool { + for _, item := range slice { + if item == sliceItem { + return true + } + } + return false +} + +func getStorageAnnotation(annotations map[string]string) (string, bool) { + value, ok := annotations[Config.storageAnnotationKey] + if ok { + return value, true + } + return "", false +} + +func hasStorageDefined(monitor monitoringv1.PodMonitor) bool { + _, ok := getStorageAnnotation(monitor.ObjectMeta.Annotations) + return ok +} + +func (r *PodMonitorReconciler) getNamespace(namespacename string) (*corev1.Namespace, error) { + ns := corev1.Namespace{} + + if err := r.Get(context.Background(), types.NamespacedName{Name: namespacename}, &ns); err != nil { + r.Log.Error(err, "Namespace could not be fetched") + return nil, err + } + + return &ns, nil +} + +func customRemoteWriteSecretName(secret corev1.Secret) string { + return fmt.Sprintf("custom-remote-write-%s-%s", secret.Name, secret.Namespace) +} + +// Returns a namespacedname referencing a configMap, bool whether we could retrieve all the data, and error if any +func (r *PodMonitorReconciler) getNamespacedNameObjectFromAnnotation(monitor monitoringv1.PodMonitor, annotation string) (*types.NamespacedName, bool, error) { + ns, err := r.getNamespace(monitor.Namespace) + if err != nil { + return nil, false, err + } + + customAlertManagerConfigMap, ok := ns.ObjectMeta.Annotations[annotation] + if ok { + parts := strings.Split(customAlertManagerConfigMap, "/") + namespacedName := types.NamespacedName{} + if len(parts) == 2 { //we have name and namespace + namespacedName.Name = parts[1] + namespacedName.Namespace = parts[0] + } else { + namespacedName.Name = parts[0] + namespacedName.Namespace = monitor.Namespace + } + return &namespacedName, true, nil + } + return nil, false, nil +} + +func (r *PodMonitorReconciler) reconcilePodMonitor(log logr.Logger, monitor monitoringv1.PodMonitor) error { + prometheus := newPrometheusObjectDef(monitor.Namespace, r.PrometheusNamespace) + log = log.WithValues("prometheusNamespace", prometheus.Namespace, "prometheusName", prometheus.Name) + + if !hasStorageDefined(monitor) { + log.Info("Skipping PodMonitor with no storage") + return nil + } + + grafanaStackTenantMapping := map[string]*grafanacloud.Stack{} + if r.EnableMetricsRemoteWrite && isGrafanaCloudStorageEnabled(monitor.ObjectMeta) { + stacks, err := lookupGrafanaStacks(r.Client, monitor.Namespace) + if err != nil { + return err + } + + for _, stack := range stacks { + grafanaCloudStack, err := r.GrafanaCloudClient.GetStack(stack) + if err != nil { + log.Error(err, fmt.Sprintf("Skipping PodMonitor for stack '%s'. Check error message", stack)) + return err + } + + secret := &corev1.Secret{} + if err := r.Client.Get(context.Background(), client.ObjectKey{ + Namespace: r.PrometheusNamespace, + Name: r.GrafanaCloudCredentials, + }, secret); err != nil { + log.Error(err, fmt.Sprintf("secret %s not found. The secret is needed to configure the remote write to Grafana Cloud.", r.GrafanaCloudCredentials)) + os.Exit(1) + } + + secret.Data[stack] = []byte(fmt.Sprintf("%d", grafanaCloudStack.MetricsInstanceID)) + if err := r.Client.Update(context.Background(), secret); err != nil { + log.Error(err, fmt.Sprintf("secret %s could not be updated. The secret is needed to configure the remote write to Grafana Cloud.", r.GrafanaCloudCredentials)) + return err + } + + log.WithValues("grafanacloudTenant", stack, "grafanaCloudStack", grafanaCloudStack.StackID).Info("injected grafana cloud credentials") + grafanaStackTenantMapping[stack] = grafanaCloudStack + } + } + if secret, ok, _ := r.getNamespacedNameObjectFromAnnotation(monitor, Config.remoteWriteAnnotationKey); ok { + err := r.syncCustomStorageSecret(secret, log, monitor) + if err != nil { + return err + } + } + err := r.createOrUpdatePrometheus(monitor, prometheus, grafanaStackTenantMapping, r.NodeSelectorTarget) + if err != nil { + log.Error(err, "Can not create or update Prometheus.") + return err + } + log.Info("created or updated prometheus") + + err = r.createOrUpdateAdditionalScrappingConfiguration(prometheus) + if err != nil { + log.Error(err, "Can not create or update the Secret with the additional scrapping configuration.") + return err + } + log.Info("created or updated secret with additional scrapping config") + err = r.createOrUpdatePrometheusRules(monitor, prometheus) + if err != nil { + log.Error(err, "Can not create or update PromethesRules.") + return err + } + log.Info("created or updated prometheus rules") + + if r.EnableVpa { + err = r.createOrUpdateVerticalPodAutoscaler(prometheus) + if err != nil { + log.Error(err, "Can not create or update VerticalPodAutoscaler.") + return err + } + log.Info("created or updated prometheus VPA") + } + + return nil +} + +func (r *PodMonitorReconciler) syncSecret(secretName, secretNamespace, targetNamespace string) error { + srcSecret := corev1.Secret{} + ctx := context.Background() + + if err := r.Get(ctx, types.NamespacedName{Namespace: secretNamespace, Name: secretName}, &srcSecret); err != nil { + r.Log.Error(err, "Error: secret %s not found. The secret is needed it to configure the remote write to the custom remote storage.", secretName) + return err + } + + dstSecret := corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: customRemoteWriteSecretName(srcSecret), + Namespace: targetNamespace, + }, + } + + _, err := ctrl.CreateOrUpdate(ctx, r.Client, &dstSecret, func() error { + dstSecret.Data = srcSecret.Data + return nil + }) + return err + +} + +func (r *PodMonitorReconciler) syncCustomStorageSecret(namespacedName *types.NamespacedName, log logr.Logger, monitor monitoringv1.PodMonitor) error { + secret := corev1.Secret{} + ctx := context.Background() + + if err := r.Get(ctx, *namespacedName, &secret); err != nil { + log.Error(err, "Error: secret %s not found. The secret is needed it to configure the remote write to the custom remote storage.", *namespacedName) + return err + } + + if referencedSecrets, ok := secret.ObjectMeta.Annotations[Config.referencedSecretAnnotationKeys]; ok { + for _, ref := range strings.Split(referencedSecrets, ",") { + err := r.syncSecret(ref, secret.Namespace, r.PrometheusNamespace) + if err != nil { + return err + } + + } + } + return nil +} + +func getAvailableStorageFromNamespace(k8sclient client.Client, namespace string, log logr.Logger) (string, error) { + ns := corev1.Namespace{} + err := k8sclient.Get(context.TODO(), types.NamespacedName{Name: namespace}, &ns) + if err != nil { + log.Error(err, "Failed to get Namespace") + return "", err + } + + availableStorage, ok := getStorageAnnotation(ns.ObjectMeta.Annotations) + if ok { + log.Info("using storage from namespace annotation: " + availableStorage) + return availableStorage, nil + } + log.Info("using default storage: grafanacloud") + + return "grafanacloud", nil +} + +func (r *PodMonitorReconciler) remoteWriteConfigFromSecret(secretRef types.NamespacedName) ([]monitoringv1.RemoteWriteSpec, error) { + secret := corev1.Secret{} + err := r.Get(context.TODO(), secretRef, &secret) + if err != nil { + r.Log.Error(err, "Failed to get secret %s", secretRef) + return nil, err + } + spec := monitoringv1.RemoteWriteSpec{} + remoteWrite, ok := secret.Data["remote-write"] + if !ok { + return nil, fmt.Errorf("'remote-write' entry doest not exist in %s", secretRef) + + } + var body interface{} + err = yaml.Unmarshal(remoteWrite, &body) + if err != nil { + return nil, err + } + jsonspec, err := json.Marshal(body) + if err != nil { + return nil, err + } + err = json.Unmarshal(jsonspec, &spec) + if err != nil { + return nil, err + } + + if spec.URL == "" { + return nil, fmt.Errorf("Missing URL in %s", string(jsonspec)) + } + + if spec.BasicAuth != nil { + s, err := getSecret(r.Client, types.NamespacedName{Name: spec.BasicAuth.Password.Name, Namespace: secretRef.Namespace}) + if err != nil { + return nil, err + } + spec.BasicAuth.Password.Name = customRemoteWriteSecretName(*s) + s, err = getSecret(r.Client, types.NamespacedName{Name: spec.BasicAuth.Username.Name, Namespace: secretRef.Namespace}) + if err != nil { + return nil, err + } + spec.BasicAuth.Username.Name = customRemoteWriteSecretName(*s) + } + + return []monitoringv1.RemoteWriteSpec{spec}, nil + +} + +func (r *PodMonitorReconciler) alertManagerConfigFromConfigMap(configMapRef types.NamespacedName) (*monitoringv1.AlertingSpec, error) { + configMap := corev1.ConfigMap{} + err := r.Get(context.TODO(), configMapRef, &configMap) + if err != nil { + r.Log.Error(err, "Failed to get configmap %s", configMapRef) + return nil, err + } + spec := monitoringv1.AlertingSpec{} + + alertManager, ok := configMap.Data["alert-manager"] + if !ok { + return nil, fmt.Errorf("'alert-manager' entry doest not exist in %s", configMapRef) + + } + + var body interface{} + err = yaml.Unmarshal([]byte(alertManager), &body) + if err != nil { + return nil, err + } + jsonspec, err := json.Marshal(body) + if err != nil { + return nil, err + } + err = json.Unmarshal(jsonspec, &spec.Alertmanagers) + if err != nil { + return nil, err + } + + return &spec, nil + +} + +func (r *PodMonitorReconciler) publishErrEvent(name string, podmonitor monitoringv1.PodMonitor, e error) error { + event := corev1.Event{ + ObjectMeta: metav1.ObjectMeta{ + Name: "GrafanaCloudOperatorEvent", + Namespace: podmonitor.Namespace, + }, + } + + _, err := ctrl.CreateOrUpdate( + context.TODO(), + r.Client, + &event, + func() error { + event.Message = e.Error() + event.Count = event.Count + 1 + event.LastTimestamp = metav1.NewTime(time.Now()) + event.InvolvedObject = corev1.ObjectReference{ + Kind: podmonitor.Kind, + Namespace: podmonitor.Namespace, + Name: podmonitor.Name, + } + return nil + }, + ) + return err + +} + +func (r *PodMonitorReconciler) getTenantCustomRelabelConfigs(namespace string, prometheus *monitoringv1.Prometheus) []monitoringv1.RelabelConfig { + var existingTenantCustomRelabelConfigs []monitoringv1.RelabelConfig + defaultPrometheusConfig := monitoringv1.RelabelConfig{ + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + } + configMap := corev1.ConfigMap{} + + if len(prometheus.Spec.RemoteWrite) > 0 { + existingTenantCustomRelabelConfigs = prometheus.Spec.RemoteWrite[0].WriteRelabelConfigs + } else { + existingTenantCustomRelabelConfigs = append(existingTenantCustomRelabelConfigs, defaultPrometheusConfig) + } + + err := r.Client.Get(context.TODO(), client.ObjectKey{Namespace: namespace, Name: "custom-relabel-configs"}, &configMap) + if err != nil { + if k8sErrors.IsNotFound(err) { + r.Log.Info("ConfigMap not found, returning empty relabel configs") + return existingTenantCustomRelabelConfigs + } + + r.Log.Error(err, "Failed to get ConfigMap %s", "custom-relabel-configs") + return existingTenantCustomRelabelConfigs + } + + relabelConfigs, ok := configMap.Data["relabel-configs"] + if !ok { + r.Log.Error(err, "Failed to get ConfigMap key relabel-configs", "custom-relabel-configs") + + publishErr := publishPrometheusErrEvent(r.Client, "CustomRelabelConfigReadError", namespace, prometheus, err) + if publishErr != nil { + r.Log.Error(publishErr, "Error sending error Event to K8s") + } + + return existingTenantCustomRelabelConfigs + } + + var configs []monitoringv1.RelabelConfig + err = yaml.Unmarshal([]byte(relabelConfigs), &configs) + if err != nil { + r.Log.Error(err, "Failed to unmarshal relabel-configs") + + publishErr := publishPrometheusErrEvent(r.Client, "CustomRelabelConfigUnmarshalError", namespace, prometheus, err) + if publishErr != nil { + r.Log.Error(publishErr, "Error sending error Event to K8s") + } + + return existingTenantCustomRelabelConfigs + } + + r.Log.Info(fmt.Sprintf("Successfully fetched Custom RelabelConfig ConfigMap Name for %s", namespace)) + + configs = append(configs, defaultPrometheusConfig) + + return configs +} + +func (r *PodMonitorReconciler) updatePrometheusObjectDef(monitor monitoringv1.PodMonitor, prometheus *monitoringv1.Prometheus, grafanaStackTenantMapping map[string]*grafanacloud.Stack, nodeSelectorTarget string) error { + replicas := int32(1) + shards := int32(1) + // It is important not to overwrite the whole ObjectMeta in order not to break + // the CreateOrUpdate function + var availableStorage string + availableStorage, err := getAvailableStorageFromNamespace(r.Client, monitor.Namespace, r.Log) + if err != nil { + return err + } + + prometheus.ObjectMeta.Annotations = map[string]string{ + Config.storageAnnotationKey: availableStorage, + Config.accountAnnotationKey: monitor.Namespace, + } + // If there were resources set up manually we keep them + // This is now required because big namespaces like serenity need big prometheus + // but we dont have a way (yet) to tell how much resources a given prometheus needs. + // if parent object doesnt have any resource, we set a default one. + resources := prometheus.Spec.Resources + if resources.Size() == 0 { + resources = defaultResourceRequirements + } + + var remoteWriteConfig []monitoringv1.RemoteWriteSpec + if r.EnableMetricsRemoteWrite && isGrafanaCloudStorageEnabled(monitor.ObjectMeta) { + for stack, grafanaCloudStack := range grafanaStackTenantMapping { + remoteWriteConfig = append(remoteWriteConfig, monitoringv1.RemoteWriteSpec{ + URL: fmt.Sprintf("%s/api/prom/push", grafanaCloudStack.PromURL), + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: r.GrafanaCloudCredentials, + }, + Key: stack, + }, + Password: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: r.GrafanaCloudCredentials, + }, + Key: "grafana-cloud-api-key", + }, + }, + WriteRelabelConfigs: r.getTenantCustomRelabelConfigs(monitor.Namespace, prometheus), + }) + } + } + + if secret, ok, _ := r.getNamespacedNameObjectFromAnnotation(monitor, Config.remoteWriteAnnotationKey); ok { + remoteWriteConfig, err = r.remoteWriteConfigFromSecret(*secret) + if err != nil { + publishErr := r.publishErrEvent("RemoteWriteConfigFromSecret", monitor, err) + if publishErr != nil { + r.Log.Error(publishErr, "Error sending error Event to K8s") + } + return err + } + } + var alertingConfig *monitoringv1.AlertingSpec + if alertingConfigMap, ok, _ := r.getNamespacedNameObjectFromAnnotation(monitor, Config.alertManagerAnnotationKey); ok { + alertingConfig, err = r.alertManagerConfigFromConfigMap(*alertingConfigMap) + if err != nil { + publishErr := r.publishErrEvent("alertManagerconfigFromConfigMap", monitor, err) + if publishErr != nil { + r.Log.Error(publishErr, "Error sending Event to K8s") + } + return err + } + } + mergedLabels := map[string]string{ + "cluster": r.ClusterName, + "monitor": "prometheus-local", + "account": monitor.Namespace, + "region": r.Region, + } + + for k, v := range r.PrometheusExtraExternalLabels { + mergedLabels[k] = v + } + + prometheus.Spec = monitoringv1.PrometheusSpec{ + ListenLocal: false, + Paused: false, + PortName: "web", + PriorityClassName: r.PrometheusPodPriorityClassName, + RoutePrefix: "/", + Retention: "30m", + ServiceAccountName: r.PrometheusServiceAccountName, + LogFormat: "logfmt", + LogLevel: "warn", + Replicas: &replicas, + Shards: &shards, + Version: r.PrometheusDockerImage.Tag, + BaseImage: r.PrometheusDockerImage.Name, + ExternalURL: "http://" + monitor.Namespace + "-metrics." + r.PrometheusExposedDomain, + RemoteWrite: remoteWriteConfig, + Alerting: alertingConfig, + ExternalLabels: mergedLabels, + PodMetadata: &monitoringv1.EmbeddedObjectMetadata{ + Labels: map[string]string{ + Config.accountLabelKey: monitor.Namespace, + }, + }, + PodMonitorNamespaceSelector: &metav1.LabelSelector{}, + PodMonitorSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + Config.accountLabelKey: monitor.Namespace, + }, + }, + RuleNamespaceSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "name": monitor.Namespace, + }, + }, + RuleSelector: &metav1.LabelSelector{}, + ServiceMonitorNamespaceSelector: &metav1.LabelSelector{}, + ServiceMonitorSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + Config.accountLabelKey: monitor.Namespace, + }, + }, + Resources: resources, + } + + if r.PrometheusMonitoringTarget != "" { + prometheus.Spec.AdditionalScrapeConfigs = &corev1.SecretKeySelector{ + Key: "prometheus", + LocalObjectReference: corev1.LocalObjectReference{Name: r.secretName(prometheus, r.PrometheusMonitoringTarget)}, + } + } + + if nodeSelectorTarget != "" { + prometheus.Spec.NodeSelector = map[string]string{ + nodeSelectorTarget: "true", + } + prometheus.Spec.Tolerations = []corev1.Toleration{ + { + Key: nodeSelectorTarget, + Operator: "Equal", + Value: "true", + Effect: "NoSchedule", + }, + } + } + return nil +} + +func (r *PodMonitorReconciler) createOrUpdateAdditionalScrappingConfig(prometheus *monitoringv1.Prometheus, monitoringTargetName string) error { + ctx := context.Background() + + additionalScrapingConfigSecret := r.newAdditionalScrapingSecretDef(prometheus, monitoringTargetName) + + _, err := ctrl.CreateOrUpdate( + ctx, + r.Client, + additionalScrapingConfigSecret, + func() error { + err := r.updateAdditionalScrappingSecretDef(prometheus, additionalScrapingConfigSecret, monitoringTargetName) + if err != nil { + r.Log.Error(err, "Failed to update secret with additional scrapping config") + prometheusErrors.WithLabelValues(prometheus.Name, prometheus.Namespace, additionalScrapingConfigSecret.Kind).Inc() + return err + } + + err = controllerutil.SetOwnerReference(prometheus, additionalScrapingConfigSecret, r.Scheme) + if err != nil { + r.Log.Error(err, "Failed to assign owner reference to the secret with additional scrapping config") + prometheusErrors.WithLabelValues(prometheus.Name, prometheus.Namespace, additionalScrapingConfigSecret.Kind).Inc() + err = fmt.Errorf("%v %v", additionalScrapingConfigSecret, err) + return err + } + + return nil + }, + ) + return err +} + +func (r *PodMonitorReconciler) createOrUpdatePrometheusRules(monitor monitoringv1.PodMonitor, prometheus *monitoringv1.Prometheus) error { + ruleName := "prometheus-remote-write-behind-seconds" + rules := []monitoringv1.Rule{ + { + Record: strings.ReplaceAll(ruleName, "-", "_"), + Expr: intstr.FromString( + "max_over_time(prometheus_remote_storage_highest_timestamp_in_seconds{job=\"prometheus-scraper\"}[2m]) " + + "- ignoring(remote_name, url) group_right " + + "max_over_time(prometheus_remote_storage_queue_highest_sent_timestamp_seconds{job=\"prometheus-scraper\"}[2m])"), + }, + } + err := r.createOrUpdateInternalPrometheusRule(prometheus, ruleName, rules) + if err != nil { + r.Log.Error(err, "Can not create or update a prometheus rule", "prometheus", prometheus.Name, "prometheusRule", ruleName) + return err + } + if monitor.Spec.SampleLimit > 0 { + ruleName = "scrape-config-sample-limit" + rules = []monitoringv1.Rule{ + { + Record: strings.ReplaceAll(ruleName, "-", "_"), + Expr: intstr.FromInt(int(monitor.Spec.SampleLimit)), + Labels: map[string]string{ + "job": fmt.Sprintf("%s/%s", monitor.Namespace, monitor.Name), + "namespace": monitor.Namespace, + }, + }, + } + err = r.createOrUpdatePrometheusRule(&monitor, fmt.Sprintf("%s-%s", monitor.GetName(), ruleName), rules) + if err != nil { + r.Log.Error(err, "Can not create or update a prometheus rule", "prometheus", prometheus.Name, "prometheusRule", ruleName) + return err + } + } + + ruleName = "prometheus-remote-write-storage-failures-percentage" + rules = []monitoringv1.Rule{ + { + Record: strings.ReplaceAll(ruleName, "-", "_"), + Expr: intstr.FromString("(rate(prometheus_remote_storage_failed_samples_total{job=\"prometheus-scraper\"}[2m])/(rate(prometheus_remote_storage_failed_samples_total{job=\"prometheus-scraper\"}[2m])+rate(prometheus_remote_storage_succeeded_samples_total{job=\"prometheus-scraper\"}[2m])))* 100"), + }, + } + err = r.createOrUpdateInternalPrometheusRule(prometheus, ruleName, rules) + if err != nil { + r.Log.Error(err, "Can not create or update a prometheus rule", "prometheus", prometheus.Name, "prometheusRule", ruleName) + return err + } + return nil +} + +func (r *PodMonitorReconciler) createOrUpdateInternalPrometheusRule(owner metav1.Object, name string, rules []monitoringv1.Rule) error { + return r.createOrUpdatePrometheusRule(owner, fmt.Sprintf("%s-%s", getObjectName(owner), name), rules) +} + +func (r *PodMonitorReconciler) createOrUpdatePrometheusRule(owner metav1.Object, name string, rules []monitoringv1.Rule) error { + ctx := context.Background() + + prometheusRule := r.newPrometheusRuleObjectDef(owner.GetNamespace(), name) + + _, err := ctrl.CreateOrUpdate( + ctx, + r.Client, + prometheusRule, + func() error { + err := r.updatePrometheusRuleObjectDef(owner, prometheusRule, rules) + if err != nil { + r.Log.Error(err, "Failed to update prometheus rule") + prometheusErrors.WithLabelValues(owner.GetName(), owner.GetNamespace(), prometheusRule.Kind).Inc() + return err + } + + err = controllerutil.SetOwnerReference(owner, prometheusRule, r.Scheme) + if err != nil { + r.Log.Error(err, "Failed to assign owner reference to prometheus rule") + prometheusErrors.WithLabelValues(owner.GetName(), owner.GetNamespace(), prometheusRule.Kind).Inc() + return err + } + + return nil + }, + ) + return err +} + +func (r *PodMonitorReconciler) newPrometheusRuleObjectDef(namespace, name string) *monitoringv1.PrometheusRule { + return &monitoringv1.PrometheusRule{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: namespace, + }, + } +} +func (r *PodMonitorReconciler) updatePrometheusRuleObjectDef(owner metav1.Object, prometheusRule *monitoringv1.PrometheusRule, rules []monitoringv1.Rule) error { + + tenantNamespace, found := owner.GetAnnotations()[Config.accountAnnotationKey] + if !found { + return errors.New("annotation" + Config.accountAnnotationKey + " not found in owner Object.") + } + + prometheusRule.ObjectMeta.Labels = map[string]string{ + Config.accountLabelKey: tenantNamespace, + } + prometheusRule.Spec = monitoringv1.PrometheusRuleSpec{ + Groups: []monitoringv1.RuleGroup{ + { + Name: prometheusRule.ObjectMeta.Name, + Rules: rules, + }, + }, + } + return nil +} + +func (r *PodMonitorReconciler) secretName(prometheus *monitoringv1.Prometheus, targetMonitor string) string { + return fmt.Sprintf("%s-%s", getObjectName(prometheus), targetMonitor) +} + +func (r *PodMonitorReconciler) newAdditionalScrapingSecretDef(prometheus *monitoringv1.Prometheus, targetMonitor string) *corev1.Secret { + return &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: r.secretName(prometheus, targetMonitor), + Namespace: prometheus.Namespace, + }, + } + +} + +func (r *PodMonitorReconciler) updateAdditionalScrappingSecretDef(prometheus *monitoringv1.Prometheus, cm *corev1.Secret, targetMonitor string) error { + tenantNamespace, found := prometheus.Spec.PodMetadata.Labels[Config.accountLabelKey] + if !found { + return errors.New("Pod metadata label " + Config.accountLabelKey + " not found in Prometheus.") + } + cm.ObjectMeta.Labels = map[string]string{ + Config.accountLabelKey: tenantNamespace, + } + + additionalScrapeConfig := PrometheusAdditionalScrapeConfig{} + scrapper := newIngressAndClusterScraper() + scrapper.JobName = fmt.Sprintf("%s/prometheus-%s-%s/0", prometheus.Namespace, tenantNamespace, targetMonitor) + scrapper.ServiceDiscoveryConfigs = append(scrapper.ServiceDiscoveryConfigs, &kubernetes.SDConfig{ + Role: kubernetes.RoleService, + NamespaceDiscovery: kubernetes.NamespaceDiscovery{ + Names: []string{prometheus.Namespace}, + }, + HTTPClientConfig: promcommonconfig.DefaultHTTPClientConfig, + }) + scrapper.Params.Add("match[]", fmt.Sprintf(`{federate="true", namespace="%s"}`, tenantNamespace)) + + additionalScrapeConfig = append(additionalScrapeConfig, &prometheusLocalScrapper) + additionalScrapeConfig = append(additionalScrapeConfig, &scrapper) + + data, err := yaml.Marshal(additionalScrapeConfig) + if err != nil { + return fmt.Errorf("There was an error Marshalling the additional scrapping config %v", err) + } + + cm.StringData = map[string]string{ + "prometheus": fixPrometheusConfig(string(data)), + } + + return nil +} + +func fixPrometheusConfig(cfg string) string { + redirectRe := regexp.MustCompile("[ ]*follow_redirects:[ ]+(false|true)") + + kubeconfigFileRe := regexp.MustCompile(`[ ]*kubeconfig_file:[ ]+""`) + // For some reason, the prometheus serializer generates an additional follow_reditect: false + // at the root of the kubernetes service discovery while it's set to true inside + cfg = redirectRe.ReplaceAllString(cfg, "") + cfg = kubeconfigFileRe.ReplaceAllString(cfg, "") + return cfg +} + +func (r *PodMonitorReconciler) createOrUpdateAdditionalScrappingConfiguration(prometheus *monitoringv1.Prometheus) error { + targetName := r.PrometheusMonitoringTarget + err := r.createOrUpdateAdditionalScrappingConfig(prometheus, targetName) + if err != nil { + r.Log.Error(err, "Can not create or update the secret with additional scrapping config", "prometheus", prometheus.Name, "secretName", targetName) + return err + } + + return nil +} + +func (r *PodMonitorReconciler) removeFinalizer(podMonitor *monitoringv1.PodMonitor) error { + controllerutil.RemoveFinalizer(podMonitor, Config.podmonitorFinalizer) + if err := r.Update(context.Background(), podMonitor); err != nil { + return err + } + + if err := r.deletePrometheusIfNotRequired(*podMonitor); err != nil { + return err + } + return nil +} +func (r *PodMonitorReconciler) handleFinalizers(podMonitor *monitoringv1.PodMonitor) (bool, error) { + + podMonitorWithFinalizer := controllerutil.ContainsFinalizer(podMonitor, Config.podmonitorFinalizer) + // examine DeletionTimestamp to determine if object is under deletion + if podMonitor.ObjectMeta.DeletionTimestamp.IsZero() { + // The object is not being deleted, so if it does not have our finalizer, + // then lets add the finalizer and update the object. This is equivalent + // registering our finalizer. + if !podMonitorWithFinalizer { + podMonitor.ObjectMeta.Finalizers = append(podMonitor.ObjectMeta.Finalizers, Config.podmonitorFinalizer) + if err := r.Update(context.Background(), podMonitor); err != nil { + return true, err + } + } + return false, nil + } + if podMonitorWithFinalizer { + if err := r.removeFinalizer(podMonitor); err != nil { + return true, err + } + } + return true, nil +} + +func (r *PodMonitorReconciler) createOrUpdatePrometheus(monitor monitoringv1.PodMonitor, prometheus *monitoringv1.Prometheus, grafanaStackTenantMapping map[string]*grafanacloud.Stack, nodeSelectorTarget string) error { + ctx := context.Background() + _, err := ctrl.CreateOrUpdate( + ctx, + r.Client, + prometheus, + func() error { + return r.updatePrometheusObjectDef(monitor, prometheus, grafanaStackTenantMapping, nodeSelectorTarget) + }, + ) + if err != nil { + prometheusErrors.WithLabelValues(prometheus.Name, prometheus.Namespace, prometheus.Kind).Inc() + return err + } + + return nil +} + +func (r *PodMonitorReconciler) createOrUpdateVerticalPodAutoscaler(prometheus *monitoringv1.Prometheus) error { + updateModeAuto := vpav1.UpdateMode("Auto") + vpa := &vpav1.VerticalPodAutoscaler{ObjectMeta: metav1.ObjectMeta{Name: prometheus.Name, Namespace: prometheus.Namespace}} + + ctx := context.Background() + _, err := ctrl.CreateOrUpdate( + ctx, + r.Client, + vpa, + func() error { + vpa.Spec = vpav1.VerticalPodAutoscalerSpec{ + UpdatePolicy: &vpav1.PodUpdatePolicy{ + UpdateMode: &updateModeAuto, + }, + ResourcePolicy: &vpav1.PodResourcePolicy{ + ContainerPolicies: []vpav1.ContainerResourcePolicy{{ + ContainerName: "prometheus", + MaxAllowed: corev1.ResourceList{ + corev1.ResourceCPU: *resource.NewQuantity(8, resource.BinarySI), + }, + }}}, + } + + targetRef := &autoscaling.CrossVersionObjectReference{ + APIVersion: monitoringv1.SchemeGroupVersion.String(), + Kind: "Prometheus", + Name: prometheus.Name, + } + vpa.Spec.TargetRef = targetRef + err := controllerutil.SetOwnerReference(prometheus, vpa, r.Scheme) + return err + }, + ) + return err +} + +func (r *PodMonitorReconciler) SetupWithManager(mgr ctrl.Manager) error { + return ctrl.NewControllerManagedBy(mgr). + For(&monitoringv1.PodMonitor{}). + Complete(r) +} diff --git a/pkg/controllers/podmonitor_controller_test.go b/pkg/controllers/podmonitor_controller_test.go new file mode 100644 index 0000000..bd9e588 --- /dev/null +++ b/pkg/controllers/podmonitor_controller_test.go @@ -0,0 +1,1933 @@ +package controllers + +import ( + "context" + "fmt" + "math/rand" + "os" + "testing" + "time" + + "github.com/adevinta/observability-operator/pkg/grafanacloud" + "github.com/adevinta/observability-operator/pkg/test_helpers" + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "gopkg.in/yaml.v3" + corev1 "k8s.io/api/core/v1" + k8serrors "k8s.io/apimachinery/pkg/api/errors" + "k8s.io/apimachinery/pkg/api/resource" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/runtime" + "k8s.io/apimachinery/pkg/runtime/schema" + "k8s.io/apimachinery/pkg/types" + "k8s.io/apimachinery/pkg/util/intstr" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +func newNamespace(name string) *corev1.Namespace { + return &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + }, + } +} + +func newDefaultTestPodMonitorReconciler(t *testing.T, initialObjects ...runtime.Object) *PodMonitorReconciler { + t.Helper() + scheme := runtime.NewScheme() + _ = monitoringv1.AddToScheme(scheme) + os.Setenv("GRAFANA_CLOUD_TOKEN", "zzzz") + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + return nil, nil + }, + } + + fakeClient := test_helpers.NewFakeClient(t, initialObjects...) + return &PodMonitorReconciler{ + Client: fakeClient, + Region: "eu-fake-1", + PrometheusDockerImage: DockerImage{ + Name: "foo", + Tag: "v0.0.1", + }, + PrometheusNamespace: "prometheusesNamespace", + PrometheusExposedDomain: "clustername.clusterdomain", + ClusterName: "clustername", + Log: ctrl.Log.WithName("controllers").WithName("Deployment"), + Scheme: scheme, + GrafanaCloudCredentials: "observability-operator-grafana-cloud-credentials", + GrafanaCloudClient: &grafanaCloudClient, + PrometheusServiceAccountName: "prometheus-tenant", + PrometheusMonitoringTarget: "monitoring-target", + PrometheusExtraExternalLabels: map[string]string{ + "mycustomkey": "mycustomvalue", + }, + } +} + +func createCustomStorageSecretStub(namespace string) (*corev1.Secret, *corev1.Secret, *corev1.Secret) { + remoteWrite := ` + basicAuth: + password: + key: password + name: secret2 + username: + key: username + name: secret1 + url: https://victoria-metrics-url` + + remoteWriteSecret := &corev1.Secret{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "secret-storage", + Namespace: namespace, + Annotations: map[string]string{ + Config.referencedSecretAnnotationKeys: "secret1,secret2", + }, + }, + Data: map[string][]uint8{ + "remote-write": []byte(remoteWrite), + }, + } + secret1 := &corev1.Secret{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "secret1", + Namespace: namespace, + }, + Data: map[string][]uint8{ + "username": []byte("foo"), + }, + } + + secret2 := &corev1.Secret{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "secret2", + Namespace: namespace, + }, + Data: map[string][]uint8{ + "password": []byte("bar"), + }, + } + + return remoteWriteSecret, secret1, secret2 +} + +func createSecretStub(namespace string) *corev1.Secret { + return &corev1.Secret{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "observability-operator-grafana-cloud-credentials", + Namespace: namespace, + }, + Data: map[string][]uint8{ + "grafana-cloud-api-key": []byte("xxx"), + "grafana-cloud-traces-token": []byte("yyy"), + }, + } +} + +func getScrappingConfig() (PrometheusAdditionalScrapeConfig, error) { + var scrap PrometheusAdditionalScrapeConfig + b, err := os.ReadFile("fixtures/additional_scrapping_config") + if err != nil { + return scrap, err + } + err = yaml.Unmarshal(b, &scrap) + if err != nil { + return scrap, err + } + return scrap, err +} + +func createPodStub(name, namespace string) *corev1.Pod { + return &corev1.Pod{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: namespace, + Labels: map[string]string{"app": name}, + }, + Spec: corev1.PodSpec{}, + } +} + +func createPodMonitor(namespace string, name string) *monitoringv1.PodMonitor { + return &monitoringv1.PodMonitor{ + ObjectMeta: ctrl.ObjectMeta{ + Namespace: namespace, + Name: name, + }, + } +} + +func createPrometheusObject(namespace string, name string) *monitoringv1.Prometheus { + replicas := int32(1) + shards := int32(1) + return &monitoringv1.Prometheus{ + ObjectMeta: ctrl.ObjectMeta{ + Name: name, + Namespace: namespace, + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.accountAnnotationKey: name, + }, + }, + Spec: monitoringv1.PrometheusSpec{ + AdditionalScrapeConfigs: &corev1.SecretKeySelector{ + Key: "prometheus", + LocalObjectReference: corev1.LocalObjectReference{Name: fmt.Sprintf("prometheus-%s-monitoring-target", name)}, + }, + PodMetadata: &monitoringv1.EmbeddedObjectMetadata{ + Labels: map[string]string{ + Config.accountLabelKey: name, + }, + }, + ServiceMonitorSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{Config.accountLabelKey: name}, + }, + PodMonitorSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{Config.accountLabelKey: name}, + }, + RuleNamespaceSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "name": name, + }, + }, + ServiceMonitorNamespaceSelector: &metav1.LabelSelector{}, + PodMonitorNamespaceSelector: &metav1.LabelSelector{}, + RoutePrefix: "/", + ServiceAccountName: "prometheus-tenant", + PortName: "web", + Version: "v0.0.1", + BaseImage: "foo", + Replicas: &replicas, + Shards: &shards, + LogLevel: "warn", + LogFormat: "logfmt", + Retention: "30m", + Resources: defaultResourceRequirements, + ExternalLabels: map[string]string{ + "cluster": "clustername", + "region": "eu-fake-1", + "account": name, + "monitor": "prometheus-local", + "mycustomkey": "mycustomvalue", + }, + ExternalURL: fmt.Sprintf("http://%s-metrics.clustername.clusterdomain", name), + RemoteWrite: []monitoringv1.RemoteWriteSpec{{ + URL: "https://prometheus-us-central1.grafana.net/api/prom/push", + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "adevintaruntime", + }, + Password: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "grafana-cloud-api-key", + }, + }, + WriteRelabelConfigs: []monitoringv1.RelabelConfig{ + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + }, + }}, + RuleSelector: &metav1.LabelSelector{}, + }, + } +} + +func createIngressServiceMonitor() *monitoringv1.ServiceMonitor { + return &monitoringv1.ServiceMonitor{ + ObjectMeta: metav1.ObjectMeta{ + Name: "prometheus-podmonitorNamespace-adevinta-ingress-metrics", + Namespace: "prometheusesNamespace", + Labels: map[string]string{ + Config.accountLabelKey: "podmonitorNamespace", + }, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "monitoring.coreos.com/v1", + Kind: "Prometheus", + Name: "podmonitorNamespace", + UID: "", + Controller: nil, + BlockOwnerDeletion: nil, + }, + }, + }, + Spec: monitoringv1.ServiceMonitorSpec{ + Endpoints: []monitoringv1.Endpoint{ + { + Interval: "30s", + Params: map[string][]string{ + "match[]": {"{federate=\"true\", namespace=\"podmonitorNamespace\"}"}, + }, + Path: "/federate", + RelabelConfigs: []*monitoringv1.RelabelConfig{ + { + Action: "labeldrop", + Regex: "pod", + }, + { + Action: "labeldrop", + Regex: "node", + }, + { + Action: "labeldrop", + Regex: "namespace", + }, + }, + MetricRelabelConfigs: []*monitoringv1.RelabelConfig{ + { + Action: "labeldrop", + Regex: "federate", + }, + }, + HonorLabels: true, + }, + }, + Selector: metav1.LabelSelector{ + MatchLabels: map[string]string{ + "release": "adevinta-ingress-metrics", + }, + }, + }, + } +} + +func createRemoteWriteBehindSecondsPrometheusRule() *monitoringv1.PrometheusRule { + return &monitoringv1.PrometheusRule{ + ObjectMeta: metav1.ObjectMeta{ + Name: "prometheus-podmonitorNamespace-prometheus-remote-write-behind-seconds", + Namespace: "prometheusesNamespace", + Labels: map[string]string{ + Config.accountLabelKey: "podmonitorNamespace", + }, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "monitoring.coreos.com/v1", + Kind: "Prometheus", + Name: "podmonitorNamespace", + UID: "", + Controller: nil, + BlockOwnerDeletion: nil, + }, + }, + }, + Spec: monitoringv1.PrometheusRuleSpec{ + Groups: []monitoringv1.RuleGroup{ + { + Name: "prometheus-podmonitorNamespace-prometheus-remote-write-behind-seconds", + Rules: []monitoringv1.Rule{ + { + Record: "prometheus_remote_write_behind_seconds", + Expr: intstr.FromString( + "max_over_time(prometheus_remote_storage_highest_timestamp_in_seconds{job=\"prometheus-scraper\"}[2m]) " + + "- ignoring(remote_name, url) group_right " + + "max_over_time(prometheus_remote_storage_queue_highest_sent_timestamp_seconds{job=\"prometheus-scraper\"}[2m])"), + }, + }, + }, + }, + }, + } +} + +func createRemoteWriteStorageFailuresPercentPrometheusRule() *monitoringv1.PrometheusRule { + return &monitoringv1.PrometheusRule{ + ObjectMeta: metav1.ObjectMeta{ + Name: "prometheus-podmonitorNamespace-prometheus-remote-write-storage-failures-percentage", + Namespace: "prometheusesNamespace", + Labels: map[string]string{ + Config.accountLabelKey: "podmonitorNamespace", + }, + OwnerReferences: []metav1.OwnerReference{ + { + APIVersion: "monitoring.coreos.com/v1", + Kind: "Prometheus", + Name: "podmonitorNamespace", + UID: "", + Controller: nil, + BlockOwnerDeletion: nil, + }, + }, + }, + Spec: monitoringv1.PrometheusRuleSpec{ + Groups: []monitoringv1.RuleGroup{ + { + Name: "prometheus-podmonitorNamespace-prometheus-remote-write-storage-failures-percentage", + Rules: []monitoringv1.Rule{ + { + Record: "prometheus_remote_write_storage_failures_percentage", + Expr: intstr.FromString("(rate(prometheus_remote_storage_failed_samples_total{job=\"prometheus-scraper\"}[2m])/(rate(prometheus_remote_storage_failed_samples_total{job=\"prometheus-scraper\"}[2m])+rate(prometheus_remote_storage_succeeded_samples_total{job=\"prometheus-scraper\"}[2m])))* 100"), + }, + }, + }, + }, + }, + } +} + +func createPrometheusRulesWithoutSampleLimit() []*monitoringv1.PrometheusRule { + return []*monitoringv1.PrometheusRule{ + createRemoteWriteBehindSecondsPrometheusRule(), + createRemoteWriteStorageFailuresPercentPrometheusRule(), + } +} + +func getRecordingRuleByRecordName(prometheusRules []*monitoringv1.PrometheusRule, recordName string) monitoringv1.Rule { + for _, promRules := range prometheusRules { + for _, groups := range promRules.Spec.Groups { + for _, rule := range groups.Rules { + if rule.Record == recordName { + return rule + } + } + } + } + return monitoringv1.Rule{} +} + +func TestPodMonitorWithoutStorageDoesNotCreatePrometheus(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + reconciler := newDefaultTestPodMonitorReconciler(t, podmonitor) + reconciler.EnableMetricsRemoteWrite = true + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + require.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + assert.Len(t, prometheuses.Items, 0) +} + +func TestPodMonitorWithoutStorageDoesNotCreatePrometheusRules(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + reconciler := newDefaultTestPodMonitorReconciler(t, podmonitor) + reconciler.EnableMetricsRemoteWrite = true + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + require.NotNil(t, result) + prometheusRules := &monitoringv1.PrometheusRuleList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheusRules)) + assert.Len(t, prometheusRules.Items, 0) +} + +func TestPodMonitorWithoutStorageDoesNotCreateIngressServiceMonitor(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + reconciler := newDefaultTestPodMonitorReconciler(t, podmonitor) + reconciler.EnableMetricsRemoteWrite = true + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + require.NotNil(t, result) + serviceMonitors := &monitoringv1.ServiceMonitorList{} + + require.NoError(t, reconciler.Client.List(context.Background(), serviceMonitors)) + assert.Len(t, serviceMonitors.Items, 0) +} + +func TestPodMonitorWithGrafanaCloudStorageCreatesPrometheus(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + t.Run("it should create the prometheus object", func(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + ) + reconciler.PrometheusPodPriorityClassName = "prometheus-priority-class" + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + expectedPrometheus.Spec.PriorityClassName = "prometheus-priority-class" + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + assert.Equal(t, *expectedPrometheus, *prometheuses.Items[0]) + }) + + t.Run("it should fail if the grafana API returns a non-200 listing the stacks.", func(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + return nil, fmt.Errorf("not found") + }, + } + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + newNamespace(podmonitor.Namespace), + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + // Declaring the []monitoringv1.RemoteWriteSpec slice at assignment makes the value nil instead of an empty slice. + // -> var remoteWriteConfig []monitoringv1.RemoteWriteSpec creates an empty slice. + // -> expectedPrometheus.Spec.RemoteWrite = []monitoringv1.RemoteWriteSpec{} creates a nil slice. + // The test expects an empty slice. + var remoteWriteConfig []monitoringv1.RemoteWriteSpec + expectedPrometheus.Spec.RemoteWrite = remoteWriteConfig + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + assert.Error(t, err) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + assert.Equal(t, len(prometheuses.Items), 0) + }) + + t.Run("if a Prometheus existed, the remoteWriteconfig should be kept when the grafana API fails listing the stacks with a non-200 return code.", func(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + return nil, fmt.Errorf("not found") + }, + } + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + + prom := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + prom.Spec.RemoteWrite = []monitoringv1.RemoteWriteSpec{{ + Name: "a-test-connection-remote-write", + URL: "grafana-api-to-send-labels.com", + }} + expectedRemoteWrite := prom.Spec.RemoteWrite + + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + newNamespace(podmonitor.Namespace), + prom, + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + _, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + assert.Error(t, err) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + assert.Equal(t, len(prometheuses.Items), 1) + assert.Equal(t, expectedRemoteWrite, prometheuses.Items[0].Spec.RemoteWrite) + }) +} + +func TestPodMonitorWithCustomRelabelConfigs(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + t.Run("it should create the prometheus object with a relabel config", func(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + createConfigMapRulesStub("custom-relabel-configs", podmonitor.Namespace), + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + expectedPrometheus.Spec.RemoteWrite[0].WriteRelabelConfigs = []monitoringv1.RelabelConfig{ + { + Action: "labeldrop", + Regex: "^prometheus$", + }, + { + Action: "labeldrop", + Regex: "^node_exporter$", + }, + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + } + assert.Equal(t, *expectedPrometheus, *prometheuses.Items[0]) + }) + + t.Run("it should not update the prometheus object with the previous relabel config if the new configMap is wrong", func(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + existingPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + existingPrometheus.Spec.RemoteWrite[0].WriteRelabelConfigs = []monitoringv1.RelabelConfig{ + { + Action: "labeldrop", + SourceLabels: []string{"__name__"}, + Regex: "^prometheus$", + }, + } + + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + createConfigMapBrokenRulesStub("custom-relabel-configs", podmonitor.Namespace), + existingPrometheus, + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + expectedPrometheus.Spec.RemoteWrite[0].WriteRelabelConfigs = []monitoringv1.RelabelConfig{ + { + Action: "labeldrop", + SourceLabels: []string{"__name__"}, + Regex: "^prometheus$", + }, + } + assert.Equal(t, expectedPrometheus.Spec.RemoteWrite[0].WriteRelabelConfigs, prometheuses.Items[0].Spec.RemoteWrite[0].WriteRelabelConfigs) + }) + + t.Run("it should create the prometheus object without errors if no relabel config is defined", func(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + + // ns.Annotations[stackNameAnnotationKey] = "adevintastack1" + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + assert.Equal(t, *expectedPrometheus, *prometheuses.Items[0]) + }) + + t.Run("it should create the prometheus object and don't remove the remoteWriteConfig if configMap info is wrong", func(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + createConfigMapBrokenRulesStub("custom-relabel-configs", podmonitor.Namespace), + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + expectedPrometheus.Spec.RemoteWrite[0].WriteRelabelConfigs = []monitoringv1.RelabelConfig{ + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + } + assert.Equal(t, *expectedPrometheus, *prometheuses.Items[0]) + }) +} + +func TestPodMonitorWithGrafanaCloudStorageAndMultipleStacksCreatesPrometheus(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + switch tenant { + case "adevintastack1": + return &grafanacloud.Stack{ + Slug: "adevintastack1", + MetricsInstanceID: 14111, + PromURL: "https://prometheus-us-west1.grafana.net", + }, nil + case "adevintastack2": + return &grafanacloud.Stack{ + Slug: "adevintastack2", + MetricsInstanceID: 14818, + PromURL: "https://prometheus-us-central1.grafana.net", + }, nil + case "stack3": + return &grafanacloud.Stack{ + Slug: "stack3", + MetricsInstanceID: 14820, + PromURL: "https://prometheus-us-east1.grafana.net", + }, nil + } + + stack := grafanacloud.Stack{} + return &stack, nil + }, + } + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintastack1,adevintastack2,stack3", + }) + + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.RemoteWrite = []monitoringv1.RemoteWriteSpec{ + { + URL: "https://prometheus-us-west1.grafana.net/api/prom/push", + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "adevintastack1", + }, + Password: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "grafana-cloud-api-key", + }, + }, + WriteRelabelConfigs: []monitoringv1.RelabelConfig{ + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + }, + }, + { + URL: "https://prometheus-us-central1.grafana.net/api/prom/push", + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "adevintastack2", + }, + Password: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "grafana-cloud-api-key", + }, + }, + WriteRelabelConfigs: []monitoringv1.RelabelConfig{ + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + }, + }, + { + URL: "https://prometheus-us-east1.grafana.net/api/prom/push", + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "stack3", + }, + Password: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "grafana-cloud-api-key", + }, + }, + WriteRelabelConfigs: []monitoringv1.RelabelConfig{ + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + }, + }, + } + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + assert.ElementsMatch(t, expectedPrometheus.Spec.RemoteWrite, prometheuses.Items[0].Spec.RemoteWrite) +} + +func TestCreatesPrometheusWithSelectorAndToleration(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + + reconciler := newDefaultTestPodMonitorReconciler( + t, + podmonitor, createSecretStub("prometheusesNamespace"), + newNamespace("prometheusesNamespace"), + podMonitorNamespace, + ) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusesNamespace" + reconciler.NodeSelectorTarget = "my-dedicated-prometheus-pool" + reconciler.GrafanaCloudClient = &grafanaCloudClient + expectedPrometheus := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + expectedPrometheus.Spec.Tolerations = []corev1.Toleration{ + { + Key: reconciler.NodeSelectorTarget, + Operator: "Equal", + Value: "true", + Effect: "NoSchedule", + }, + } + expectedPrometheus.Spec.NodeSelector = map[string]string{ + reconciler.NodeSelectorTarget: "true", + } + assert.Equal(t, *expectedPrometheus, *prometheuses.Items[0]) +} + +func TestPodMonitorWithGrafanaCloudStorageDoesNotCreateServiceMonitor(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + secret := createSecretStub("prometheusNamespace") + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + reconciler := newDefaultTestPodMonitorReconciler(t, secret, podmonitor, podMonitorNamespace) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = "prometheusNamespace" + reconciler.GrafanaCloudClient = &grafanaCloudClient + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + assert.Len(t, prometheuses.Items, 1) + + serviceMonitors := &monitoringv1.ServiceMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), serviceMonitors)) + assert.Len(t, serviceMonitors.Items, 0) +} + +func TestPodMonitorWithGrafanaCloudStorageCreatesASecretWithAdditionalScrapingConfig(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + promNamespace := "prometheusesNamespace" + secret := createSecretStub(promNamespace) + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + expectedIngressServiceMonitor := createIngressServiceMonitor() + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, expectedIngressServiceMonitor, podmonitor, podMonitorNamespace) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = promNamespace + reconciler.GrafanaCloudClient = &grafanaCloudClient + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + assert.NotNil(t, result) + targetScrappingConfig, err := getScrappingConfig() + require.NoError(t, err) + + prometheusConfigSecret := &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: "prometheusesNamespace", + Name: "prometheus-podmonitorNamespace-monitoring-target", + }, + } + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(prometheusConfigSecret), prometheusConfigSecret)) + actualScrappingConfig := PrometheusAdditionalScrapeConfig{} + err = yaml.Unmarshal([]byte(prometheusConfigSecret.StringData["prometheus"]), &actualScrappingConfig) + require.Nil(t, err) + assert.Equal(t, targetScrappingConfig, actualScrappingConfig) +} + +func TestPodMonitorWithGrafanaCloudStorageCreatesTwoPrometheusRulesIfNoSampleLimit(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + promNamespace := "prometheusesNamespace" + secret := createSecretStub(promNamespace) + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.accountAnnotationKey: "podmonitorNamespace", + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + expectedRules := createPrometheusRulesWithoutSampleLimit() + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, podmonitor, podMonitorNamespace) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = promNamespace + reconciler.GrafanaCloudClient = &grafanaCloudClient + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + + promRules := monitoringv1.PrometheusRuleList{} + require.NoError(t, reconciler.Client.List(context.Background(), &promRules)) + for i, rule := range promRules.Items { + if i < len(expectedRules) { + expectedRules[i].ResourceVersion = rule.ResourceVersion + } + } + assert.Equal(t, expectedRules, promRules.Items) +} + +func TestPodMonitorWithGrafanaCloudStorageCreatesThreePrometheusRulesIfSampleLimitIsSet(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + promNamespace := "promNamespace" + secret := createSecretStub(promNamespace) + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + sampleLimit := 10 + rand.Intn(50000) + podmonitor.Spec.SampleLimit = uint64(sampleLimit) + podmonitor.Annotations = map[string]string{ + Config.accountAnnotationKey: "podmonitorNamespace", + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, podmonitor, podMonitorNamespace) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = promNamespace + reconciler.GrafanaCloudClient = &grafanaCloudClient + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + + promRules := monitoringv1.PrometheusRuleList{} + require.NoError(t, reconciler.Client.List(context.Background(), &promRules)) + + assert.Equal(t, intstr.FromInt(sampleLimit), getRecordingRuleByRecordName(promRules.Items, "scrape_config_sample_limit").Expr) +} + +func TestTwoPodMonitorInTheSameNamespaceOnlyCreateOnePrometheus(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + promNamespace := "prometheusNamespace" + secret := createSecretStub(promNamespace) + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + podmonitor2 := podmonitor.DeepCopy() + podmonitor2.Name = "podmonitorName2" + + expectedPrometheus := createPrometheusObject(promNamespace, "podmonitorNamespace") + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, podmonitor, podmonitor2, podMonitorNamespace) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = promNamespace + reconciler.GrafanaCloudClient = &grafanaCloudClient + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + require.NotNil(t, result) + + // The first reconcile should work to provide the expected context for the actual test case + result, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor2.Namespace, Name: podmonitor2.Name}}) + require.NoError(t, err) + require.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + assert.NotNil(t, prometheuses.Items[0]) + expectedPrometheus.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + expectedPrometheus.ResourceVersion = prometheuses.Items[0].ResourceVersion + assert.Equal(t, *expectedPrometheus, *prometheuses.Items[0]) + + podmonitors := &monitoringv1.PodMonitorList{} + require.NoError(t, reconciler.Client.List(context.Background(), podmonitors)) + require.Len(t, podmonitors.Items, 2) +} + +func TestReconcileHonoursResources(t *testing.T) { + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + prom := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + secret := createSecretStub(prom.Namespace) + somegigabytes, _ := resource.ParseQuantity("2Gi") + somecpu, _ := resource.ParseQuantity("1500m") + resources := &corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + "cpu": somecpu, + "memory": somegigabytes, + }, + Requests: corev1.ResourceList{ + "cpu": somecpu, + "memory": somegigabytes, + }, + } + prom.Spec.Resources = *resources + expectedPortName := prom.Spec.PortName + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podMonitorNamespace := newNamespace(podmonitor.Namespace) + podMonitorNamespace.SetAnnotations(map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }) + + prom.Spec.PortName = "wrongPortName" + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, prom, podmonitor, newNamespace(prom.Namespace), podMonitorNamespace) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = prom.Namespace + reconciler.GrafanaCloudClient = &grafanaCloudClient + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + require.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + assert.NotNil(t, prometheuses.Items[0]) + + assert.Equal(t, expectedPortName, prometheuses.Items[0].Spec.PortName) + assert.Equal(t, *resources, prometheuses.Items[0].Spec.Resources) +} + +func TestDeleteOnePodMonitorDoesNotDeletePrometheus(t *testing.T) { + prom := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + secret := createSecretStub(prom.Namespace) + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podmonitor2 := podmonitor.DeepCopy() + podmonitor2.Name = "podmonitorName2" + now := metav1.NewTime(time.Now()) + + podmonitor.DeletionTimestamp = &now + podmonitor.Finalizers = append(podmonitor.Finalizers, Config.podmonitorFinalizer) + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, prom, podmonitor, podmonitor2, newNamespace(prom.Namespace)) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = prom.Namespace + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + assert.NotNil(t, prometheuses.Items[0]) + prom.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + require.Nil(t, err) + require.NotNil(t, result) + assert.Equal(t, *prom, *prometheuses.Items[0]) +} + +func TestDeleteLastPodMonitorDeletesPrometheus(t *testing.T) { + prom := createPrometheusObject("prometheusesNamespace", "podmonitorNamespace") + + secret := createSecretStub(prom.Namespace) + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + now := metav1.NewTime(time.Now()) + + podmonitor.ObjectMeta.Finalizers = append(podmonitor.ObjectMeta.Finalizers, Config.podmonitorFinalizer) + podmonitor.DeletionTimestamp = &now + + reconciler := newDefaultTestPodMonitorReconciler(t, secret, prom, podmonitor, newNamespace(prom.Namespace)) + reconciler.EnableMetricsRemoteWrite = true + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Nil(t, err) + require.NotNil(t, result) + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + assert.Len(t, prometheuses.Items, 0) +} + +func TestPodMonitorDeFederateNamespace(t *testing.T) { + // The result should be a Prometheus Object whose storage is only `grafanacloud` + // Even though EnableFederation = true + ns := &corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "podmonitorNamespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.stackNameAnnotationKey: "adevintaruntime", + }, + }, + } + + var replicas int32 = 1 + var shards int32 = 1 + prom := &monitoringv1.Prometheus{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "podmonitorNamespace", + Namespace: "prometheusesNamespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.accountAnnotationKey: "podmonitorNamespace", + }, + }, + Spec: monitoringv1.PrometheusSpec{ + AdditionalScrapeConfigs: &corev1.SecretKeySelector{ + Key: "prometheus", + LocalObjectReference: corev1.LocalObjectReference{Name: "prometheus-podmonitorNamespace-monitoring-target"}, + }, + PodMetadata: &monitoringv1.EmbeddedObjectMetadata{ + Labels: map[string]string{ + Config.accountLabelKey: "podmonitorNamespace", + }, + }, + ServiceMonitorSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{Config.accountLabelKey: "podmonitorNamespace"}, + }, + PodMonitorSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{Config.accountLabelKey: "podmonitorNamespace"}, + }, + RuleNamespaceSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "name": "podmonitorNamespace", + }, + }, + ServiceMonitorNamespaceSelector: &metav1.LabelSelector{}, + PodMonitorNamespaceSelector: &metav1.LabelSelector{}, + RoutePrefix: "/", + ServiceAccountName: "prometheus-tenant", + PortName: "web", + Version: "v0.0.1", + BaseImage: "foo", + Replicas: &replicas, + Shards: &shards, + LogLevel: "warn", + LogFormat: "logfmt", + Retention: "30m", + Resources: defaultResourceRequirements, + ExternalLabels: map[string]string{ + "cluster": "clustername", + "region": "eu-fake-1", + "account": "podmonitorNamespace", + "monitor": "prometheus-local", + "mycustomkey": "mycustomvalue", + }, + ExternalURL: "http://podmonitorNamespace-metrics.clustername.clusterdomain", + RemoteWrite: []monitoringv1.RemoteWriteSpec{{ + URL: "https://prometheus-us-central1.grafana.net/api/prom/push", + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "adevintaruntime", + }, + Password: corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: "observability-operator-grafana-cloud-credentials", + }, + Key: "grafana-cloud-api-key", + }, + }, + WriteRelabelConfigs: []monitoringv1.RelabelConfig{ + { + Action: "drop", + SourceLabels: []string{"__name__"}, + Regex: "prometheus_.*", + }, + }, + }}, + RuleSelector: &metav1.LabelSelector{}, + }, + } + + podmonitor := monitoringv1.PodMonitor{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "podmonitorName", + Namespace: "podmonitorNamespace", + Labels: map[string]string{ + Config.podMonitorLabelKey: "podmonitorNamespace-podmonitorName", + Config.accountLabelKey: "podmonitorNamespace", + }, + }, + Spec: monitoringv1.PodMonitorSpec{ + PodMetricsEndpoints: []monitoringv1.PodMetricsEndpoint{ + { + Path: "/metrics", + HonorLabels: true, + }, + }, + Selector: metav1.LabelSelector{ + MatchLabels: map[string]string{ + "random_label": "true", + }, + }, + NamespaceSelector: monitoringv1.NamespaceSelector{MatchNames: []string{"bar"}}, + SampleLimit: 1500, + }, + } + secret := createSecretStub(prom.Namespace) + + grafanaCloudClient := mockGrafanaCloudClient{ + GetStackFunc: func(tenant string) (*grafanacloud.Stack, error) { + stack := grafanacloud.Stack{ + PromURL: "https://prometheus-us-central1.grafana.net", + } + return &stack, nil + }, + } + + reconciler := newDefaultTestPodMonitorReconciler(t, ns, &podmonitor, secret, prom) + reconciler.EnableMetricsRemoteWrite = true + reconciler.PrometheusNamespace = prom.Namespace + reconciler.GrafanaCloudClient = &grafanaCloudClient + + annotation := map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.accountAnnotationKey: "accountName", + } + + podmonitor.ObjectMeta.Annotations = annotation + require.NoError(t, reconciler.Client.Update(context.Background(), &podmonitor)) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + require.NotNil(t, prometheuses.Items[0]) + + prom.Spec.PodMetadata.Annotations = prometheuses.Items[0].Spec.PodMetadata.Annotations + prom.ResourceVersion = prometheuses.Items[0].ResourceVersion + prom.TypeMeta = prometheuses.Items[0].TypeMeta + assert.Equal(t, *prom, *prometheuses.Items[0]) +} + +func TestPodMonitorWithAlternativeStorageEnabled(t *testing.T) { + secret := &corev1.Secret{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "observability-operator-grafana-cloud-credentials", + Namespace: "platform-services", + }, + Data: map[string][]uint8{ + "grafana-cloud-api-key": []byte("xxx"), + }, + } + podmonitor := monitoringv1.PodMonitor{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "podmonitorName", + Namespace: "podmonitorNamespace", + Labels: map[string]string{ + Config.podMonitorLabelKey: "podmonitorNamespace-podmonitorName", + Config.accountLabelKey: "podmonitorNamespace", + }, + Annotations: map[string]string{ + Config.storageAnnotationKey: "federation", + Config.accountAnnotationKey: "podmonitorNamespace", + }, + }, + Spec: monitoringv1.PodMonitorSpec{ + PodMetricsEndpoints: []monitoringv1.PodMetricsEndpoint{ + { + Path: "/metrics", + HonorLabels: true, + }, + }, + Selector: metav1.LabelSelector{ + MatchLabels: map[string]string{ + "random_label": "true", + }, + }, + NamespaceSelector: monitoringv1.NamespaceSelector{MatchNames: []string{"bar"}}, + SampleLimit: 1500, + }, + } + ns := newNamespace(podmonitor.Namespace) + ns.Annotations = map[string]string{ + Config.storageAnnotationKey: "federation", + } + reconciler := newDefaultTestPodMonitorReconciler(t, &podmonitor, secret, ns) + reconciler.EnableMetricsRemoteWrite = false + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + assert.Nil(t, err) + assert.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + assert.Len(t, prometheuses.Items, 1) + assert.Empty(t, prometheuses.Items[0].Spec.RemoteWrite) +} + +func TestVpaEnabledCreatesVpaObject(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podmonitor.Name = "podmonitorName" + podmonitor.Namespace = "my-podmonitor-namespace-pro" + reconciler := newDefaultTestPodMonitorReconciler(t, podmonitor, newNamespace(podmonitor.Namespace)) + reconciler.EnableVpa = true + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + + vpas := &vpav1.VerticalPodAutoscalerList{} + require.NoError(t, reconciler.Client.List(context.Background(), vpas)) + require.Len(t, vpas.Items, 1) + require.NotNil(t, vpas.Items[0].Spec.UpdatePolicy.UpdateMode) + assert.Equal(t, vpav1.UpdateMode("Auto"), *vpas.Items[0].Spec.UpdatePolicy.UpdateMode) + assert.Equal(t, podmonitor.Namespace, vpas.Items[0].Name) + assert.Equal(t, reconciler.PrometheusNamespace, vpas.Items[0].Namespace) + require.Len(t, vpas.Items[0].OwnerReferences, 1) + assert.Equal(t, monitoringv1.SchemeGroupVersion.Identifier(), vpas.Items[0].OwnerReferences[0].APIVersion) + assert.Equal(t, "Prometheus", vpas.Items[0].OwnerReferences[0].Kind) + assert.Equal(t, "my-podmonitor-namespace-pro", vpas.Items[0].OwnerReferences[0].Name) + + result, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + + vpas = &vpav1.VerticalPodAutoscalerList{} + require.NoError(t, reconciler.Client.List(context.Background(), vpas)) + require.Len(t, vpas.Items, 1) + assert.Equal(t, "monitoring.coreos.com/v1", vpas.Items[0].Spec.TargetRef.APIVersion) + assert.Equal(t, "Prometheus", vpas.Items[0].Spec.TargetRef.Kind) + assert.Equal(t, "my-podmonitor-namespace-pro", vpas.Items[0].Spec.TargetRef.Name) + require.NotNil(t, vpas.Items[0].Spec.UpdatePolicy.UpdateMode) + assert.Equal(t, vpav1.UpdateMode("Auto"), *vpas.Items[0].Spec.UpdatePolicy.UpdateMode) + assert.Equal(t, podmonitor.Namespace, vpas.Items[0].Name) + assert.Equal(t, reconciler.PrometheusNamespace, vpas.Items[0].Namespace) + require.Len(t, vpas.Items[0].OwnerReferences, 1) + assert.Equal(t, monitoringv1.SchemeGroupVersion.Identifier(), vpas.Items[0].OwnerReferences[0].APIVersion) + assert.Equal(t, "Prometheus", vpas.Items[0].OwnerReferences[0].Kind) + assert.Equal(t, "my-podmonitor-namespace-pro", vpas.Items[0].OwnerReferences[0].Name) +} + +func TestVpaDisabledSkipsVpaObjectCreation(t *testing.T) { + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + } + podmonitor.Name = "podmonitorName" + podmonitor.Namespace = "my-podmonitor-namespace-pro" + reconciler := newDefaultTestPodMonitorReconciler(t) + reconciler.EnableVpa = false + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + assert.Nil(t, err) + assert.NotNil(t, result) + + vpas := &vpav1.VerticalPodAutoscalerList{} + require.NoError(t, reconciler.Client.List(context.Background(), vpas)) + require.Len(t, vpas.Items, 0) +} + +func TestGetAvailableStorageFromNamespace(t *testing.T) { + type TestCase struct { + Description string + Name string + Namespace corev1.Namespace + ExpectedResult string + ExpectedError error + } + + testCases := []TestCase{ + { + Description: "when the namespace exists and has the storage annotation", + Name: "mynamespace", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "mynamespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + }, + }, + }, + ExpectedResult: "grafanacloud", + ExpectedError: nil, + }, + { + Description: "when the namespace exists and does not have the storage annotation", + Name: "subito-pro", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "subito-pro", + Annotations: map[string]string{}, + }, + }, + ExpectedResult: "grafanacloud", + ExpectedError: nil, + }, + { + Description: "when the namespace exists and does not have an empty storage annotation", + Name: "some-namespace", + Namespace: corev1.Namespace{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "some-namespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "", + }, + }, + }, + ExpectedResult: "", + ExpectedError: nil, + }, + { + Description: "when the namespace does not exist an error is returned", + Name: "notfound", + Namespace: corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{Name: "other-namespace"}, + }, + ExpectedResult: "", + ExpectedError: k8serrors.NewNotFound(schema.GroupResource{Resource: "namespaces"}, "notfound"), + }, + } + for _, item := range testCases { + t.Run(item.Description, func(t *testing.T) { + reconciler := newDefaultTestPodMonitorReconciler(t, &item.Namespace) + expectedResult, expectedErr := getAvailableStorageFromNamespace(reconciler.Client, item.Name, reconciler.Log) + assert.Equal(t, item.ExpectedResult, expectedResult) + assert.Equal(t, item.ExpectedError, expectedErr) + }) + } +} + +func TestPodMonitorWithCustomStorageCreatesSecretInPromNamespaceAndRemoteWrite(t *testing.T) { + promNamespace := "prometheusesNamespace" + + ns := newNamespace("podmonitorNamespace") + ns.Annotations = map[string]string{ + Config.remoteWriteAnnotationKey: "othernamespace/secret-storage", + } + + remoteWrite, secret1, secret2 := createCustomStorageSecretStub("othernamespace") + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "custom", + } + reconciler := newDefaultTestPodMonitorReconciler(t, podmonitor, newNamespace(promNamespace), ns, newNamespace("othernamespace"), remoteWrite, secret1, secret2) + reconciler.PrometheusNamespace = promNamespace + + t.Run("Reconcile wont fail", func(t *testing.T) { + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + assert.NotNil(t, result) + }) + + t.Run("Secrets are copied to prometheus namespace", func(t *testing.T) { + createdSecret1 := &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: "prometheusesNamespace", + Name: "custom-remote-write-secret1-othernamespace", + }, + } + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(createdSecret1), createdSecret1)) + require.Equal(t, string(createdSecret1.Data["username"]), "foo") + + createdSecret2 := &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: "prometheusesNamespace", + Name: "custom-remote-write-secret2-othernamespace", + }, + } + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(createdSecret2), createdSecret2)) + require.Equal(t, string(createdSecret2.Data["password"]), "bar") + }) + + t.Run("prometheus object is created with the proper remote write ", func(t *testing.T) { + prometheus := monitoringv1.Prometheus{ + ObjectMeta: metav1.ObjectMeta{Name: "podmonitorNamespace", Namespace: promNamespace}, + } + + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(&prometheus), &prometheus)) + + require.NotEmpty(t, prometheus.Spec.RemoteWrite) + remotewriteSpec := monitoringv1.RemoteWriteSpec{ + URL: "https://victoria-metrics-url", + BasicAuth: &monitoringv1.BasicAuth{ + Username: corev1.SecretKeySelector{ + Key: "username", + LocalObjectReference: corev1.LocalObjectReference{Name: "custom-remote-write-secret1-othernamespace"}, + }, + Password: corev1.SecretKeySelector{ + Key: "password", + LocalObjectReference: corev1.LocalObjectReference{Name: "custom-remote-write-secret2-othernamespace"}, + }, + }, + } + require.Equal(t, prometheus.Spec.RemoteWrite[0], remotewriteSpec) + }) +} + +func TestPodMonitorWithCustomStorageErrors(t *testing.T) { + promNamespace := "prometheusesNamespace" + resetRenconciler := func() *PodMonitorReconciler { + ns := newNamespace("podmonitorNamespace") + ns.Annotations = map[string]string{ + Config.remoteWriteAnnotationKey: "othernamespace/secret-storage", + } + + secret1, secret2, secret3 := createCustomStorageSecretStub("othernamespace") + + podmonitor := createPodMonitor("podmonitorNamespace", "podmonitorName") + podmonitor.Annotations = map[string]string{ + Config.storageAnnotationKey: "custom", + } + reconciler := newDefaultTestPodMonitorReconciler(t, secret1, secret2, secret3, podmonitor, newNamespace(promNamespace), ns, newNamespace("othernamespace")) + reconciler.PrometheusNamespace = promNamespace + return reconciler + } + + t.Run("non existing referenced secrets will make it fail", func(t *testing.T) { + reconciler := resetRenconciler() + secret := corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: "secret-storage", + Namespace: "othernamespace", + }, + } + _, err := ctrl.CreateOrUpdate(context.Background(), reconciler.Client, &secret, func() error { + secret.Annotations[Config.referencedSecretAnnotationKeys] = "undefinedsecret" + return nil + }) + require.NoError(t, err) + + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "podmonitorNamespace", Name: "podmonitorName"}}) + require.Error(t, err) + }) + + t.Run("Non existing secret will fail make it", func(t *testing.T) { + ns := &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: "podmonitorNamespace", + }, + } + + reconciler := resetRenconciler() + _, err := ctrl.CreateOrUpdate(context.Background(), reconciler.Client, ns, func() error { + ns.Annotations[Config.remoteWriteAnnotationKey] = "notfoundsecret" + return nil + }) + require.NoError(t, err) + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "podmonitorNamespace", Name: "podmonitorName"}}) + require.Error(t, err) + }) + + t.Run("unmarshallable remote-write will make it fail and report an event", func(t *testing.T) { + reconciler := resetRenconciler() + secret := corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: "secret-storage", + Namespace: "othernamespace", + }, + } + _, err := ctrl.CreateOrUpdate(context.Background(), reconciler.Client, &secret, func() error { + secret.Data["remote-write"] = []byte("foo") + return nil + }) + require.NoError(t, err) + + _, err = reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: "podmonitorNamespace", Name: "podmonitorName"}}) + require.Error(t, err) + + event := corev1.Event{ + ObjectMeta: metav1.ObjectMeta{ + Name: "GrafanaCloudOperatorEvent", + Namespace: "podmonitorNamespace", + }, + } + err = reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(&event), &event) + require.NoError(t, err) + }) +} + +func createCustomAlertmanagerConfigMap(namespace, name, payload string) corev1.ConfigMap { + return corev1.ConfigMap{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: namespace, + }, + Data: map[string]string{ + "alert-manager": payload, + }, + } +} + +func TestPodMonitorWithAlertManager(t *testing.T) { + cmName := "alert-manager-config" + namespace := corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: "user-namespace", + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.alertManagerAnnotationKey: cmName, + }, + }, + } + podmonitor := monitoringv1.PodMonitor{ + ObjectMeta: ctrl.ObjectMeta{ + Name: "podmonitorName", + Namespace: namespace.Name, + Annotations: map[string]string{ + Config.storageAnnotationKey: "grafanacloud", + Config.alertManagerAnnotationKey: cmName, + }, + }, + } + + t.Run("Fail with invalid Alertmanager configuration", func(t *testing.T) { + cm := createCustomAlertmanagerConfigMap(namespace.Name, cmName, "is this not a valid json/yaml object?") + + reconciler := newDefaultTestPodMonitorReconciler( + t, + &podmonitor, + &namespace, + &cm, + newNamespace("prometheusesNamespace"), + ) + event := corev1.Event{ + ObjectMeta: metav1.ObjectMeta{ + Name: "GrafanaCloudOperatorEvent", + Namespace: namespace.Name, + }, + } + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.Error(t, err) + require.NotNil(t, result) + require.NoError(t, reconciler.Client.Get(context.Background(), client.ObjectKeyFromObject(&event), &event)) + }) + t.Run("Correct Alertmanager configuration in YAML", func(t *testing.T) { + cm := createCustomAlertmanagerConfigMap( + namespace.Name, + cmName, + ` + - namespace: javi-test + name: alertmanager-example-test + port: web + - namespace: javi-test + name: alertmanager-example-test-2 + port: web`, + ) + + reconciler := newDefaultTestPodMonitorReconciler( + t, + &podmonitor, + &namespace, + &cm, + newNamespace("prometheusesNamespace"), + ) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + require.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + expectedPrometheus := createPrometheusObject("prometheusesNamespace", namespace.Name) + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.Alerting = &monitoringv1.AlertingSpec{ + Alertmanagers: []monitoringv1.AlertmanagerEndpoints{ + { + Namespace: "javi-test", + Name: "alertmanager-example-test", + Port: intstr.FromString("web"), + }, + { + Namespace: "javi-test", + Name: "alertmanager-example-test-2", + Port: intstr.FromString("web"), + }, + }, + } + assert.Equal(t, *expectedPrometheus.Spec.Alerting, *prometheuses.Items[0].Spec.Alerting) + }) + t.Run("Correct Alertmanager configuration in JSON", func(t *testing.T) { + cm := createCustomAlertmanagerConfigMap( + namespace.Name, + cmName, + `[ + { + "namespace": "javi-test", + "name": "alertmanager-example-test", + "port": "web" + }, + { + "namespace": "javi-test", + "name": "alertmanager-example-test-2", + "port": "web" + } + ]`, + ) + + reconciler := newDefaultTestPodMonitorReconciler( + t, + &podmonitor, + &namespace, + &cm, + newNamespace("prometheusesNamespace"), + ) + + result, err := reconciler.Reconcile(context.Background(), ctrl.Request{NamespacedName: types.NamespacedName{Namespace: podmonitor.Namespace, Name: podmonitor.Name}}) + require.NoError(t, err) + require.NotNil(t, result) + + prometheuses := &monitoringv1.PrometheusList{} + expectedPrometheus := createPrometheusObject("prometheusesNamespace", namespace.Name) + + require.NoError(t, reconciler.Client.List(context.Background(), prometheuses)) + require.Len(t, prometheuses.Items, 1) + expectedPrometheus.Spec.Alerting = &monitoringv1.AlertingSpec{ + Alertmanagers: []monitoringv1.AlertmanagerEndpoints{ + { + Namespace: "javi-test", + Name: "alertmanager-example-test", + Port: intstr.FromString("web"), + }, + { + Namespace: "javi-test", + Name: "alertmanager-example-test-2", + Port: intstr.FromString("web"), + }, + }, + } + assert.Equal(t, *expectedPrometheus.Spec.Alerting, *prometheuses.Items[0].Spec.Alerting) + }) +} diff --git a/pkg/controllers/prometheus_job_definition.go b/pkg/controllers/prometheus_job_definition.go new file mode 100644 index 0000000..e7e9d27 --- /dev/null +++ b/pkg/controllers/prometheus_job_definition.go @@ -0,0 +1,128 @@ +package controllers + +import ( + promcommonconfig "github.com/prometheus/common/config" + "github.com/prometheus/common/model" + promconfig "github.com/prometheus/prometheus/config" + + "github.com/prometheus/prometheus/discovery" + "github.com/prometheus/prometheus/discovery/targetgroup" + "github.com/prometheus/prometheus/pkg/relabel" +) + +var prometheusLocalScrapper = promconfig.ScrapeConfig{ + JobName: "prometheus-scraper", + HonorTimestamps: true, + HTTPClientConfig: promcommonconfig.DefaultHTTPClientConfig, + ServiceDiscoveryConfigs: discovery.Configs{ + discovery.StaticConfig{ + &targetgroup.Group{ + Targets: []model.LabelSet{ + { + "__address__": "localhost:9090", + }, + }, + }, + }, + }, +} + +func newIngressAndClusterScraper() promconfig.ScrapeConfig { + duration, _ := model.ParseDuration("30s") + return promconfig.ScrapeConfig{ + HonorLabels: true, + HonorTimestamps: true, + Params: map[string][]string{ + "match[]": {}, + }, + ScrapeInterval: duration, + MetricsPath: "/federate", + ServiceDiscoveryConfigs: discovery.Configs{ + //&kubernetes.SDConfig{}, + }, + // ServiceDiscoveryConfig: sd_config.ServiceDiscoveryConfig{ + // KubernetesSDConfigs: []*kubernetes.SDConfig{}, + // }, + RelabelConfigs: []*relabel.Config{ + { + Regex: relabel.MustNewRegexp("^.*-(ingress|cluster)-metrics$"), + SourceLabels: model.LabelNames{ + "__meta_kubernetes_service_label_release", + }, + Separator: ";", + Action: relabel.Keep, + }, + { + Regex: relabel.MustNewRegexp("Node;(.*)"), + TargetLabel: "node", + Replacement: "${1}", + Separator: ";", + SourceLabels: model.LabelNames{ + "__meta_kubernetes_endpoint_address_target_kind", + "__meta_kubernetes_endpoint_address_target_name", + }, + }, + { + Regex: relabel.MustNewRegexp("Pod;(.*)"), + TargetLabel: "pod", + Replacement: "${1}", + Separator: ";", + SourceLabels: model.LabelNames{ + "__meta_kubernetes_endpoint_address_target_kind", + "__meta_kubernetes_endpoint_address_target_name", + }, + }, + { + TargetLabel: "namespace", + SourceLabels: model.LabelNames{ + "__meta_kubernetes_namespace", + }, + }, + { + TargetLabel: "service", + SourceLabels: model.LabelNames{ + "__meta_kubernetes_service_name", + }, + }, + { + TargetLabel: "pod", + SourceLabels: model.LabelNames{ + "__meta_kubernetes_pod_name", + }, + }, + { + TargetLabel: "job", + Replacement: "${1}", + SourceLabels: model.LabelNames{ + "__meta_kubernetes_service_name", + }, + }, + { + Regex: relabel.MustNewRegexp("pod"), + Action: relabel.LabelDrop, + }, + { + Regex: relabel.MustNewRegexp("node"), + Action: relabel.LabelDrop, + }, + { + Regex: relabel.MustNewRegexp("namespace"), + Action: relabel.LabelDrop, + }, + }, + MetricRelabelConfigs: []*relabel.Config{ + { + Regex: relabel.MustNewRegexp("federate"), + Action: relabel.LabelDrop, + }, + { + Regex: relabel.MustNewRegexp("__replica__"), + Action: relabel.LabelDrop, + }, + { + Regex: relabel.MustNewRegexp("^prometheus$"), + Action: relabel.LabelDrop, + }, + }, + } +} diff --git a/pkg/controllers/traces_collector.go b/pkg/controllers/traces_collector.go new file mode 100644 index 0000000..476c601 --- /dev/null +++ b/pkg/controllers/traces_collector.go @@ -0,0 +1,601 @@ +package controllers + +import ( + "bytes" + "context" + _ "embed" + "fmt" + "strconv" + "text/template" + + appsv1 "k8s.io/api/apps/v1" + autoscaling "k8s.io/api/autoscaling/v1" + corev1 "k8s.io/api/core/v1" + networkingv1 "k8s.io/api/networking/v1" + "k8s.io/apimachinery/pkg/api/resource" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/util/intstr" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" + + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +const ( + defaultObjPrefix string = "alloy-" + defaultServicePrefix string = "otelcol-" +) + +// TracesCollector abstracts the management of the Alloy service and +// its related objects. The goal is to create an Alloy deployment and +// service for each namespace that sends traces. The management of a +// single Alloy is represented by this type and the orchestration of +// the different Alloy services is managed elsewhere. +type TracesCollector struct { + Name string + HTTPPort int32 + ServicePorts []servicePort + TraceNamespace string + TenantNamespace string + ClusterName string + EnableVPA bool + + templatedConfig string + + Secret *corev1.Secret + ConfigMap *corev1.ConfigMap + Service *corev1.Service + Deployment *appsv1.Deployment + NetworkPolicy *networkingv1.NetworkPolicy + VPA *vpav1.VerticalPodAutoscaler +} + +// TracesCollectorOption implements the functional options pattern for +// TracesCollector +type TracesCollectorOption func(*TracesCollector) error + +// WithServicePorts allows to specify a list of Service Ports that +// should be enabled. +func WithServicePorts(extraPorts ...servicePort) TracesCollectorOption { + return func(tc *TracesCollector) error { + tc.ServicePorts = append(tc.ServicePorts, extraPorts...) + return nil + } +} + +// WithHTTPPort allows to specify which is the port used to expose the +// HTTP interface of Alloy +func WithHTTPPort(port int32) TracesCollectorOption { + return func(tc *TracesCollector) error { + if tc.HTTPPort != 0 { + if tc.HTTPPort == port { + // nothing to do + return nil + } + + // Remove prior port from existing ServicePort list if it was present + var newServicePorts []servicePort + for i, v := range tc.ServicePorts { + if v.Port == tc.HTTPPort { + newServicePorts = append(tc.ServicePorts[:i], tc.ServicePorts[i+1:]...) + } + } + tc.ServicePorts = newServicePorts + } + + tc.HTTPPort = port + tc.ServicePorts = append(tc.ServicePorts, servicePort{Name: "http", Port: httpPort}) + return nil + } +} + +// AlloyWithOTelCredentials renders the Alloy configuration file using +// the supplied credentials to set up the destination of traces +func AlloyWithOTelCredentials(credentials ...OTelCredentials) TracesCollectorOption { + return func(tc *TracesCollector) error { + templatedConfig, err := tc.getAlloyConfigContents(credentials...) + if err != nil { + return err + } + tc.templatedConfig = templatedConfig + return nil + } +} + +// NewTracesCollector initializes the TracesCollector type. +func NewTracesCollector(tenantNamespace, traceNamespace, clusterName string, enableVPA bool, opts ...TracesCollectorOption) (*TracesCollector, error) { + tc := &TracesCollector{ + Name: tenantNamespace, + HTTPPort: httpPort, + ServicePorts: []servicePort{{Name: "http", Port: httpPort}}, + TraceNamespace: traceNamespace, + TenantNamespace: tenantNamespace, + ClusterName: clusterName, + EnableVPA: enableVPA, + } + for _, opt := range opts { + err := opt(tc) + if err != nil { + return nil, err + } + } + return tc, nil +} + +func (tc *TracesCollector) objName() string { + return defaultObjPrefix + tc.Name +} + +func (tc *TracesCollector) svcName() string { + return defaultServicePrefix + tc.Name +} + +func (tc *TracesCollector) labelMap() map[string]string { + return map[string]string{ + "app.kubernetes.io/name": tc.objName(), + "app.kubernetes.io/managed-by": "grafana-cloud-operator", + "app.kubernetes.io/version": "v1.2.1", + } +} + +func (tc *TracesCollector) annotationMap() map[string]string { + return map[string]string{ + "prometheus.io/path": "/metrics", + "prometheus.io/port": strconv.Itoa(int(tc.HTTPPort)), + "prometheus.io/scrape": "true", + } +} + +// defineVPA returns a default initialized VPA object and +// populates it inside the TracesCollector as well +func (tc *TracesCollector) defineVPA() *vpav1.VerticalPodAutoscaler { + var updateMode vpav1.UpdateMode = "Auto" + + tc.VPA = &vpav1.VerticalPodAutoscaler{ + ObjectMeta: metav1.ObjectMeta{ + Name: tc.objName(), + Namespace: tc.TraceNamespace, + }, + Spec: vpav1.VerticalPodAutoscalerSpec{ + TargetRef: &autoscaling.CrossVersionObjectReference{ + Kind: "Deployment", + Name: tc.objName(), + }, + UpdatePolicy: &vpav1.PodUpdatePolicy{ + UpdateMode: &updateMode, + }, + }, + } + return tc.VPA +} + +// CreateOrUpdateVPA creates or updates the VPA by calling K8s API +func (tc *TracesCollector) CreateOrUpdateVPA(ctx context.Context, client client.Client) error { + vpaMutating := tc.defineVPA().DeepCopy() + + _, err := ctrl.CreateOrUpdate( + ctx, + client, + vpaMutating, + func() error { + vpaMutating.Spec = tc.VPA.Spec + return nil + }, + ) + return err +} + +// defineSecret returns a default initialized secret object and +// populates it inside the TracesCollector as well +func (tc *TracesCollector) defineSecret() *corev1.Secret { + tc.Secret = &corev1.Secret{ + ObjectMeta: metav1.ObjectMeta{ + Name: tc.objName(), + Namespace: tc.TraceNamespace, + Labels: tc.labelMap(), + }, + Type: corev1.SecretTypeOpaque, + } + return tc.Secret +} + +// CreateOrUpdateSecret creates or updates the secret by calling K8s API +func (tc *TracesCollector) CreateOrUpdateSecret(ctx context.Context, client client.Client, grafanaCloudToken string) error { + secretMutating := tc.defineSecret().DeepCopy() + _, err := ctrl.CreateOrUpdate( + ctx, + client, + secretMutating, + func() error { + secretMutating.Data = map[string][]byte{ + "grafana-cloud-traces-token": []byte(grafanaCloudToken), + } + return nil + }, + ) + return err +} + +// defineConfigMap returns a default initialized configmap object and +// populates it inside the TracesCollector as well +func (tc *TracesCollector) defineConfigMap() *corev1.ConfigMap { + tc.ConfigMap = &corev1.ConfigMap{ + ObjectMeta: metav1.ObjectMeta{ + Name: tc.objName(), + Namespace: tc.TraceNamespace, + Labels: tc.labelMap(), + }, + Data: map[string]string{ + "config.alloy": tc.templatedConfig, + }, + } + return tc.ConfigMap +} + +// CreateOrUpdateConfigMap creates or updates the configMap by calling K8s API +func (tc *TracesCollector) CreateOrUpdateConfigMap(ctx context.Context, client client.Client) error { + configMapMutating := tc.defineConfigMap().DeepCopy() + _, err := ctrl.CreateOrUpdate( + ctx, + client, + configMapMutating, + func() error { + configMapMutating.Data = tc.ConfigMap.Data + return nil + }, + ) + return err +} + +// defineService returns a default initialized service object and +// populates it inside the TracesCollector as well +func (tc *TracesCollector) defineService() *corev1.Service { + serviceName := tc.svcName() + labels := tc.labelMap() + + var ports []corev1.ServicePort + for _, sp := range tc.ServicePorts { + ports = append( + ports, + corev1.ServicePort{ + Name: serviceName + "-" + sp.Name, + Protocol: corev1.ProtocolTCP, + Port: sp.Port, + TargetPort: intstr.IntOrString{ + IntVal: sp.Port, + }, + }, + ) + } + tc.Service = &corev1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Name: serviceName, + Namespace: tc.TraceNamespace, + Labels: labels, + }, + Spec: corev1.ServiceSpec{ + // service name is the tenant namespace so that svc endpoint looks like '.observability.svc.cluster.local' + // the service selector should target the alloy pods with name "alloy-" + Selector: map[string]string{ + "app.kubernetes.io/name": labels["app.kubernetes.io/name"], + }, + Ports: ports, + }, + } + + return tc.Service +} + +// CreateOrUpdateService creates or updates the service by calling K8s API +func (tc *TracesCollector) CreateOrUpdateService(ctx context.Context, client client.Client) error { + serviceMutating := tc.defineService().DeepCopy() + _, err := ctrl.CreateOrUpdate( + ctx, + client, + serviceMutating, + func() error { + serviceMutating.Spec.Ports = tc.Service.Spec.Ports + return nil + }, + ) + return err +} + +// defineDeployment returns a default initialized Alloy deployment +// object and populates it inside the TracesCollector as well +func (tc *TracesCollector) defineDeployment() *appsv1.Deployment { + name := tc.objName() + numReplicas := int32(1) + + CPULimit := resource.MustParse("250m") + CPURequest := resource.MustParse("50m") + + MemoryLimit := resource.MustParse("500Mi") + MemoryRequest := resource.MustParse("250Mi") + + var ports []corev1.ContainerPort + for _, sp := range tc.ServicePorts { + ports = append( + ports, + corev1.ContainerPort{ + ContainerPort: sp.Port, + Name: sp.Name, + }, + ) + } + + tc.Deployment = &appsv1.Deployment{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: tc.TraceNamespace, + Labels: tc.labelMap(), + }, + Spec: appsv1.DeploymentSpec{ + Replicas: &numReplicas, + Selector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app.kubernetes.io/name": name, + }, + }, + Template: corev1.PodTemplateSpec{ + ObjectMeta: metav1.ObjectMeta{ + Labels: tc.labelMap(), + Annotations: tc.annotationMap(), + }, + Spec: corev1.PodSpec{ + Containers: []corev1.Container{ + { + Name: name, + Args: []string{ + "run", + "/etc/alloy/config.alloy", + "--storage.path=/tmp/alloy", + fmt.Sprintf("--server.http.listen-addr=0.0.0.0:%v", tc.HTTPPort), + "--server.http.ui-path-prefix=/", + "--stability.level=generally-available", + }, + Resources: corev1.ResourceRequirements{ + Limits: corev1.ResourceList{ + corev1.ResourceCPU: CPULimit, + corev1.ResourceMemory: MemoryLimit, + }, + Requests: corev1.ResourceList{ + corev1.ResourceCPU: CPURequest, + corev1.ResourceMemory: MemoryRequest, + }, + }, + Image: "docker.io/grafana/alloy:v1.2.1", + Ports: ports, + ReadinessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/-/ready", + Port: intstr.IntOrString{IntVal: tc.HTTPPort}, + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + PeriodSeconds: 5, + TimeoutSeconds: 2, + FailureThreshold: 3, + SuccessThreshold: 1, + }, + LivenessProbe: &corev1.Probe{ + ProbeHandler: corev1.ProbeHandler{ + HTTPGet: &corev1.HTTPGetAction{ + Path: "/-/ready", + Port: intstr.IntOrString{IntVal: tc.HTTPPort}, + Scheme: corev1.URISchemeHTTP, + }, + }, + InitialDelaySeconds: 10, + PeriodSeconds: 5, + TimeoutSeconds: 2, + FailureThreshold: 3, + SuccessThreshold: 1, + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "config", + MountPath: "/etc/alloy", + }, + }, + Env: []corev1.EnvVar{ + { + Name: "GRAFANA_CLOUD_TRACES_TOKEN", + ValueFrom: &corev1.EnvVarSource{ + SecretKeyRef: &corev1.SecretKeySelector{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: tc.objName(), + }, + Key: "grafana-cloud-traces-token", + }, + }, + }, + }, + }, + { + Name: "config-reloader", + Image: "ghcr.io/jimmidyson/configmap-reload:v0.12.0", + Args: []string{ + "--volume-dir=/etc/alloy", + fmt.Sprintf("--webhook-url=http://localhost:%v/-/reload", tc.HTTPPort), + }, + VolumeMounts: []corev1.VolumeMount{ + { + Name: "config", + MountPath: "/etc/alloy", + }, + }, + }, + }, + Volumes: []corev1.Volume{ + { + Name: "config", + VolumeSource: corev1.VolumeSource{ + ConfigMap: &corev1.ConfigMapVolumeSource{ + LocalObjectReference: corev1.LocalObjectReference{ + Name: tc.objName(), + }, + }, + }, + }, + }, + }, + }, + }, + } + return tc.Deployment +} + +// CreateOrUpdateDeployment creates or updates the deployment by calling K8s API +func (tc *TracesCollector) CreateOrUpdateDeployment(ctx context.Context, client client.Client) error { + deploymentMutating := tc.defineDeployment().DeepCopy() + _, err := ctrl.CreateOrUpdate( + ctx, + client, + deploymentMutating, + func() error { + deploymentMutating.Spec = tc.Deployment.Spec + return nil + }, + ) + return err +} + +// defineNetworkPolicy returns a default initialized network policy +// object restricted to a single source namespace and populates it +// inside the TracesCollector as well +func (tc *TracesCollector) defineNetworkPolicy() *networkingv1.NetworkPolicy { + name := tc.objName() + protocol := corev1.ProtocolTCP + + var ports []networkingv1.NetworkPolicyPort + for _, port := range tc.ServicePorts { + ports = append( + ports, + networkingv1.NetworkPolicyPort{ + Protocol: &protocol, + Port: &intstr.IntOrString{IntVal: port.Port}, + }, + ) + } + + tc.NetworkPolicy = &networkingv1.NetworkPolicy{ + ObjectMeta: metav1.ObjectMeta{ + Name: name, + Namespace: tc.TraceNamespace, + Labels: tc.labelMap(), + }, + Spec: networkingv1.NetworkPolicySpec{ + PodSelector: metav1.LabelSelector{ + MatchLabels: map[string]string{ + "app.kubernetes.io/name": name, + }, + }, + PolicyTypes: []networkingv1.PolicyType{ + networkingv1.PolicyTypeIngress, + }, + Ingress: []networkingv1.NetworkPolicyIngressRule{ + { + From: []networkingv1.NetworkPolicyPeer{ + { + NamespaceSelector: &metav1.LabelSelector{ + MatchLabels: map[string]string{ + "kubernetes.io/metadata.name": tc.Name, + }, + }, + }, + }, + Ports: ports, + }, + }, + }, + } + return tc.NetworkPolicy +} + +// CreateOrUpdateNetworkPolicy creates or updates the network policy by calling K8s API +func (tc *TracesCollector) CreateOrUpdateNetworkPolicy(ctx context.Context, client client.Client) error { + networkPolicyMutating := tc.defineNetworkPolicy().DeepCopy() + _, err := ctrl.CreateOrUpdate( + ctx, + client, + networkPolicyMutating, + func() error { + networkPolicyMutating.Spec = tc.NetworkPolicy.Spec + return nil + }, + ) + return err +} + +// ObjectsToDelete returns a list of all the Alloy deployment related +// objects for deletion. The TracesColector has to be initialized the +// same way as it was when creating the initial objects for the +// deletion to succeed. +func (tc *TracesCollector) ObjectsToDelete() []client.Object { + objects := []client.Object{ + tc.defineSecret(), + tc.defineConfigMap(), + tc.defineService(), + tc.defineDeployment(), + tc.defineNetworkPolicy(), + tc.defineVPA(), + } + + if tc.EnableVPA { + objects = append(objects, tc.defineVPA()) + } + + return objects +} + +type servicePort struct { + Name string + Port int32 +} + +//go:embed files/config.alloy +var alloyConfigTemplate string + +type alloyConfig struct { + Credentials []OTelCredentials + CustomResourceAttributes map[string]string +} + +func (tc *TracesCollector) getAlloyConfigContents(credentials ...OTelCredentials) (string, error) { + // Set up default credential values + var cred []OTelCredentials + for _, v := range credentials { + if v.Password == "" { + v.Password = "env(\"GRAFANA_CLOUD_TRACES_TOKEN\")" + } + cred = append(cred, v) + } + + // Render the template + config := alloyConfig{ + Credentials: cred, + CustomResourceAttributes: map[string]string{ + "k8s.cluster.name": tc.ClusterName, + "k8s.namespace.name": tc.TenantNamespace, + }, + } + tpl := template.New("alloy-config") + tpl, err := tpl.Parse(alloyConfigTemplate) + if err != nil { + return "", err + } + + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + if err != nil { + return "", err + } + return buffer.String(), nil +} + +type OTelCredentials struct { + User, Password, Endpoint string +} diff --git a/pkg/controllers/traces_collector_test.go b/pkg/controllers/traces_collector_test.go new file mode 100644 index 0000000..23a6fb8 --- /dev/null +++ b/pkg/controllers/traces_collector_test.go @@ -0,0 +1,429 @@ +package controllers + +import ( + "bytes" + "context" + "flag" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" + "os" + "os/exec" + "strings" + "testing" + "text/template" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + appsv1 "k8s.io/api/apps/v1" + corev1 "k8s.io/api/core/v1" + v1 "k8s.io/api/core/v1" + networkingv1 "k8s.io/api/networking/v1" + "k8s.io/apimachinery/pkg/types" + "k8s.io/apimachinery/pkg/util/intstr" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +var verifyAlloyRendering = flag.Bool("verify-alloy", false, "use the alloy binary to verify rendered template correctness") + +func setupAlloyLinter(t *testing.T, buffer []byte) (*exec.Cmd, *strings.Builder) { + t.Helper() + paths := []string{ + "./alloy-linux-amd64", + "./alloy", + "alloy-linux-amd64", + "alloy", + } + + var path string + var errors []error + for _, p := range paths { + np, err := exec.LookPath(p) + if err != nil { + errors = append(errors, err) + continue + } + + path = np + break + } + if path == "" { + for _, err := range errors { + assert.NoError(t, err) + } + t.Fail() + } + cmd := exec.Command(path, "fmt") + cmd.Stdin = bytes.NewReader(buffer) + var out strings.Builder + cmd.Stdout = &out + return cmd, &out +} + +func TestAlloyConfigTemplateCorrectness(t *testing.T) { + // This test verifies that changes to the template result in valid, executable templates + path, err := os.MkdirTemp("", "alloy-lint-*") + require.NoError(t, err) + defer os.RemoveAll(path) + + tpl := template.New("alloy-config") + _, err = tpl.Parse(alloyConfigTemplate) + require.NoError(t, err, "couldn't parse Alloy configuration template successfully") + + t.Run("test alloy config file generated with a single credential is syntactically correct", func(t *testing.T) { + config := alloyConfig{ + Credentials: []OTelCredentials{ + { + User: "123", + Password: `"my-pass"`, + Endpoint: `"my.domain.tld/otlp"`, + }, + }, + } + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + require.NoError(t, err, "couldn't execute Alloy configuration template successfully") + + if *verifyAlloyRendering { + cmd, out := setupAlloyLinter(t, buffer.Bytes()) + err = cmd.Run() + require.NoError(t, err) + assert.Equal(t, buffer.String(), out.String()) + } + }) + t.Run("test alloy config file generated with multiple credentials is syntactically correct", func(t *testing.T) { + config := alloyConfig{ + Credentials: []OTelCredentials{ + { + User: "123", + Password: `"my-pass"`, + Endpoint: `"my.domain.tld/otlp"`, + }, + { + User: "456", + Password: `"my-other-pass"`, + Endpoint: `"my.other.domain.tld/otlp"`, + }, + }, + } + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + require.NoError(t, err, "couldn't execute Alloy configuration template successfully") + + if *verifyAlloyRendering { + cmd, out := setupAlloyLinter(t, buffer.Bytes()) + err = cmd.Run() + require.NoError(t, err) + assert.Equal(t, buffer.String(), out.String()) + } + }) +} + +func TestAlloyConfigSupportsMultipleDestinations(t *testing.T) { + credentials := []OTelCredentials{ + { + User: "123", + Password: `"my-pass"`, + Endpoint: `"my.domain.tld/otlp"`, + }, + { + User: "456", + Password: `"my-other-pass"`, + Endpoint: `"my.other.domain.tld/otlp"`, + }, + } + + t.Run("when there are multiple destinations all of them appear", func(t *testing.T) { + tpl := template.New("alloy-config") + _, err := tpl.Parse(alloyConfigTemplate) + require.NoError(t, err, "couldn't parse Alloy configuration template successfully") + + var buffer bytes.Buffer + err = tpl.Execute(&buffer, alloyConfig{Credentials: credentials}) + require.NoError(t, err, "couldn't execute Alloy configuration template successfully") + + assert.Contains(t, buffer.String(), "otelcol.exporter.otlphttp.default_0.input") + assert.Contains(t, buffer.String(), `otelcol.auth.basic "credentials_0"`) + assert.Contains(t, buffer.String(), "username = "+credentials[0].User) + assert.Contains(t, buffer.String(), "password = "+credentials[0].Password) + assert.Contains(t, buffer.String(), `otelcol.exporter.otlphttp "default_0"`) + assert.Contains(t, buffer.String(), "endpoint = "+credentials[0].Endpoint) + assert.Contains(t, buffer.String(), "auth = otelcol.auth.basic.credentials_0.handler") + + assert.Contains(t, buffer.String(), "otelcol.exporter.otlphttp.default_1.input") + assert.Contains(t, buffer.String(), `otelcol.auth.basic "credentials_1"`) + assert.Contains(t, buffer.String(), "username = "+credentials[1].User) + assert.Contains(t, buffer.String(), "password = "+credentials[1].Password) + assert.Contains(t, buffer.String(), `otelcol.exporter.otlphttp "default_1"`) + assert.Contains(t, buffer.String(), "endpoint = "+credentials[1].Endpoint) + assert.Contains(t, buffer.String(), "auth = otelcol.auth.basic.credentials_1.handler") + }) + t.Run("when there's only one destination the others aren't there", func(t *testing.T) { + tpl := template.New("alloy-config") + _, err := tpl.Parse(alloyConfigTemplate) + require.NoError(t, err, "couldn't parse Alloy configuration template successfully") + + var buffer bytes.Buffer + err = tpl.Execute(&buffer, alloyConfig{Credentials: []OTelCredentials{credentials[0]}}) + require.NoError(t, err, "couldn't execute Alloy configuration template successfully") + + assert.Contains(t, buffer.String(), "otelcol.exporter.otlphttp.default_0.input") + assert.Contains(t, buffer.String(), `otelcol.auth.basic "credentials_0"`) + assert.Contains(t, buffer.String(), "username = "+credentials[0].User) + assert.Contains(t, buffer.String(), "password = "+credentials[0].Password) + assert.Contains(t, buffer.String(), `otelcol.exporter.otlphttp "default_0"`) + assert.Contains(t, buffer.String(), "endpoint = "+credentials[0].Endpoint) + assert.Contains(t, buffer.String(), "auth = otelcol.auth.basic.credentials_0.handler") + + assert.NotContains(t, buffer.String(), "otelcol.exporter.otlphttp.default_1.input") + assert.NotContains(t, buffer.String(), `otelcol.auth.basic "credentials_1"`) + assert.NotContains(t, buffer.String(), "username = "+credentials[1].User) + assert.NotContains(t, buffer.String(), "password = "+credentials[1].Password) + assert.NotContains(t, buffer.String(), `otelcol.exporter.otlphttp "default_1"`) + assert.NotContains(t, buffer.String(), "endpoint = "+credentials[1].Endpoint) + assert.NotContains(t, buffer.String(), "auth = otelcol.auth.basic.credentials_1.handler") + }) +} + +func TestSecretUpdate(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + gcToken := "grafana_traces_token" + gcOtherToken := "grafana_traces_token_2" + + t.Run("verify the grafana token is updated in the cluster", func(t *testing.T) { + verifySecret := func(reconciler *NamespaceReconciler, secretName, token string) { + secrets := &corev1.SecretList{} + + err := reconciler.Client.List(context.Background(), secrets, client.InNamespace(tracesNs.Name)) + require.NoError(t, err) + + require.Len(t, secrets.Items, 1) + require.Equal(t, secretName, secrets.Items[0].Name) + require.Contains(t, secrets.Items[0].Data, "grafana-cloud-traces-token") + assert.Equal(t, token, string(secrets.Items[0].Data["grafana-cloud-traces-token"])) + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs) + + tc, err := NewTracesCollector(ns.Name, tracesNs.Name, "", false) + require.NoError(t, err) + + err = tc.CreateOrUpdateSecret(context.Background(), reconciler.Client, gcToken) + require.NoError(t, err) + + verifySecret(reconciler, tc.objName(), gcToken) + + err = tc.CreateOrUpdateSecret(context.Background(), reconciler.Client, gcOtherToken) + require.NoError(t, err) + + verifySecret(reconciler, tc.objName(), gcOtherToken) + }) +} + +func TestConfigMapUpdate(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + t.Run("verify configmap is updated in the cluster", func(t *testing.T) { + verifyConfigMap := func(reconciler *NamespaceReconciler, name, config, token string) { + configmaps := &corev1.ConfigMapList{} + + err := reconciler.Client.List(context.Background(), configmaps, client.InNamespace(tracesNs.Name)) + require.NoError(t, err) + + require.Len(t, configmaps.Items, 1) + require.Equal(t, name, configmaps.Items[0].Name) + require.Contains(t, configmaps.Items[0].Data, "config.alloy") + assert.Equal(t, config, string(configmaps.Items[0].Data["config.alloy"])) + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs) + + tc, err := NewTracesCollector(ns.Name, tracesNs.Name, "", false) + require.NoError(t, err) + + err = tc.CreateOrUpdateConfigMap(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyConfigMap(reconciler, tc.objName(), "", tc.templatedConfig) + + tc.templatedConfig = "test" + err = tc.CreateOrUpdateConfigMap(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyConfigMap(reconciler, tc.objName(), "test", tc.templatedConfig) + }) +} + +func TestServiceUpdate(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + t.Run("verify service is updated in the cluster", func(t *testing.T) { + verifyService := func(reconciler *NamespaceReconciler, name string, servicePorts []servicePort) { + service := &corev1.ServiceList{} + + err := reconciler.Client.List(context.Background(), service, client.InNamespace(tracesNs.Name)) + require.NoError(t, err) + + require.Len(t, service.Items, 1) + require.Equal(t, name, service.Items[0].Name) + for _, sp := range servicePorts { + assert.Contains(t, service.Items[0].Spec.Ports, corev1.ServicePort{ + Name: name + "-" + sp.Name, + Protocol: corev1.ProtocolTCP, + Port: sp.Port, + TargetPort: intstr.IntOrString{ + IntVal: sp.Port, + }, + }) + } + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs) + + tc, err := NewTracesCollector(ns.Name, tracesNs.Name, "", false) + require.NoError(t, err) + + err = tc.CreateOrUpdateService(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyService(reconciler, tc.svcName(), []servicePort{}) + + servicePorts := []servicePort{ + { + Name: "test", + Port: int32(1234), + }, + } + tc.ServicePorts = servicePorts + + err = tc.CreateOrUpdateService(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyService(reconciler, tc.svcName(), servicePorts) + }) +} + +func TestDeploymentUpdate(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + t.Run("verify deployment is updated in the cluster", func(t *testing.T) { + verifyDeployment := func(reconciler *NamespaceReconciler, name string, servicePorts []servicePort) { + deployment := &appsv1.DeploymentList{} + + err := reconciler.Client.List(context.Background(), deployment, client.InNamespace(tracesNs.Name)) + require.NoError(t, err) + + require.Len(t, deployment.Items, 1) + require.Equal(t, name, deployment.Items[0].Name) + require.Len(t, deployment.Items[0].Spec.Template.Spec.Containers, 2) + for _, sp := range servicePorts { + assert.Contains(t, deployment.Items[0].Spec.Template.Spec.Containers[0].Ports, v1.ContainerPort{ + Name: sp.Name, + ContainerPort: sp.Port, + }) + } + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs) + + tc, err := NewTracesCollector(ns.Name, tracesNs.Name, "", false) + require.NoError(t, err) + + err = tc.CreateOrUpdateDeployment(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyDeployment(reconciler, tc.objName(), []servicePort{}) + + servicePorts := []servicePort{ + { + Name: "test", + Port: int32(1234), + }, + } + tc.ServicePorts = servicePorts + + err = tc.CreateOrUpdateDeployment(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyDeployment(reconciler, tc.objName(), servicePorts) + }) +} + +func TestNetworkPolicyUpdate(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + t.Run("verify network policy is updated in the cluster", func(t *testing.T) { + verifyNetworkPolicy := func(reconciler *NamespaceReconciler, name string, servicePorts []servicePort) { + networkPolicy := &networkingv1.NetworkPolicy{} + + err := reconciler.Client.Get(context.Background(), types.NamespacedName{Namespace: tracesNs.Name, Name: name}, networkPolicy) + require.NoError(t, err) + + if len(servicePorts) > 0 { + for _, sp := range servicePorts { + var found bool + for _, np := range networkPolicy.Spec.Ingress[0].Ports { + if sp.Port != np.Port.IntVal { + continue + } + found = true + } + assert.True(t, found) + } + } + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs) + + tc, err := NewTracesCollector(ns.Name, tracesNs.Name, "", false) + require.NoError(t, err) + + err = tc.CreateOrUpdateNetworkPolicy(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyNetworkPolicy(reconciler, tc.objName(), []servicePort{}) + + servicePorts := []servicePort{ + { + Name: "test", + Port: int32(1234), + }, + } + tc.ServicePorts = servicePorts + + err = tc.CreateOrUpdateNetworkPolicy(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyNetworkPolicy(reconciler, tc.objName(), servicePorts) + }) +} + +func TestVPAUpdate(t *testing.T) { + tracesNs := newNamespace("observability") + ns := newNamespace("my-namespace-dev") + + t.Run("verify VPA is updated in the cluster", func(t *testing.T) { + verifyVPA := func(reconciler *NamespaceReconciler, name string) { + vpa := &vpav1.VerticalPodAutoscalerList{} + + err := reconciler.Client.List(context.Background(), vpa, client.InNamespace(tracesNs.Name)) + require.NoError(t, err) + + require.Len(t, vpa.Items, 1) + require.Equal(t, name, vpa.Items[0].Name) + } + + reconciler := newDefaultNamespaceReconciler(t, ns, tracesNs) + + tc, err := NewTracesCollector(ns.Name, tracesNs.Name, "", true) + require.NoError(t, err) + + err = tc.CreateOrUpdateVPA(context.Background(), reconciler.Client) + require.NoError(t, err) + + verifyVPA(reconciler, tc.objName()) + }) +} diff --git a/pkg/grafanacloud/fluentd_template.conf b/pkg/grafanacloud/fluentd_template.conf new file mode 100644 index 0000000..01644a5 --- /dev/null +++ b/pkg/grafanacloud/fluentd_template.conf @@ -0,0 +1,101 @@ + + @type record_modifier + @id inject_loki_user_environment_dev + + environment "dev" + + + + + @type record_modifier + @id inject_loki_user_environment_pre + + environment "pre" + + + + + @type record_modifier + @id inject_loki_user_environment_pro + + environment "pro" + + + +{{- range $namespace, $creds := .Stacks }} + + @type copy +{{- range $index, $userData := $creds }} + + @type loki + @id loki_loki_{{ $namespace }}_access_log_{{ $index }} + url "{{$userData.URL}}" + username "{{ $userData.UserID }}" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 1m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "{{$.Cluster.Name}}","region":"{{$.Cluster.Region}}"} + remove_keys _sumo_metadata,source,category,host + + +{{- end }} + + + + @type copy +{{- range $index, $userData := $creds }} + + @type loki + @id loki_loki_{{ $namespace }}_{{ $index }} + url "{{$userData.URL}}" + username "{{ $userData.UserID }}" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 64m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "{{$.Cluster.Name}}","region":"{{$.Cluster.Region}}"} + remove_keys _sumo_metadata,source,category,host + + +{{- end }} + +{{- end }} diff --git a/pkg/grafanacloud/grafana_stack.go b/pkg/grafanacloud/grafana_stack.go new file mode 100644 index 0000000..95a95e9 --- /dev/null +++ b/pkg/grafanacloud/grafana_stack.go @@ -0,0 +1,86 @@ +package grafanacloud + +import ( + "time" + + "github.com/go-logr/logr" + + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "sigs.k8s.io/controller-runtime/pkg/event" +) + +type GrafanaStackReconciler struct { + Log logr.Logger + GrafanaCloudClient GrafanaCloudStackLister + GrafanaStackChangeEvents chan event.GenericEvent + OriginalStacks map[string]struct{} +} + +func (r *GrafanaStackReconciler) WatchGrafanaStacksChange(tick <-chan time.Time) error { + + stacks, err := r.listStacks() + + if err != nil { + r.Log.Error(err, "Failed to list grafana cloud stacks") + return err + } + + r.OriginalStacks = stacks + + go func() { + for range tick { + err := r.reconcile() + if err != nil { + r.Log.Error(err, "Failed to list grafana cloud stacks, it will be retried.") + } + } + }() + return nil +} + +func (r *GrafanaStackReconciler) reconcile() error { + updatedStacks, err := r.listStacks() + if err != nil { + return err + } + for updatedStack := range updatedStacks { + if _, found := r.OriginalStacks[updatedStack]; !found { + r.Log.Info("new stack is found", "stack-name", updatedStack) + r.GrafanaStackChangeEvents <- event.GenericEvent{ + Object: &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: updatedStack, + }, + }, + } + } + } + for originalStack := range r.OriginalStacks { + if _, found := updatedStacks[originalStack]; !found { + r.Log.Info("stack is deleted", "stack-name", originalStack) + r.GrafanaStackChangeEvents <- event.GenericEvent{ + Object: &corev1.Namespace{ + ObjectMeta: metav1.ObjectMeta{ + Name: originalStack, + }, + }, + } + } + } + r.OriginalStacks = updatedStacks + return nil +} + +func (r *GrafanaStackReconciler) listStacks() (map[string]struct{}, error) { + stacks, err := r.GrafanaCloudClient.ListStacks() + if err != nil { + r.Log.Error(err, "Failed to list grafana cloud stacks") + return nil, err + } + out := map[string]struct{}{} + for _, stack := range stacks { + out[stack.Slug] = struct{}{} + } + return out, nil +} diff --git a/pkg/grafanacloud/grafana_stack_test.go b/pkg/grafanacloud/grafana_stack_test.go new file mode 100644 index 0000000..8cb2852 --- /dev/null +++ b/pkg/grafanacloud/grafana_stack_test.go @@ -0,0 +1,189 @@ +package grafanacloud + +import ( + "errors" + "fmt" + "sync" + "sync/atomic" + "testing" + "time" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/event" +) + +type MockStackListerFunc struct { + ListStacksCalls int32 + ListStacksFunc func() (Stacks, error) +} + +func (cg *MockStackListerFunc) ListStacks() (Stacks, error) { + atomic.AddInt32(&cg.ListStacksCalls, 1) + if cg.ListStacksFunc != nil && cg.ListStacksCalls == 1 { + return cg.ListStacksFunc() + } + return nil, errors.New("ListStacks not implemented") +} +func TestWatchPeriodicallySyncsGrafanaCloudStacks(t *testing.T) { + grafanaStackChangeEvents := make(chan event.GenericEvent) + gcClient := &MockStackListerFunc{ + ListStacksFunc: func() (Stacks, error) { + fmt.Println("listStacks") + return []Stack{ + {Slug: "adevintaruntime", LogsInstanceID: 9876, LogsURL: "https://logs.grafanacloud.es"}, + }, nil + }, + } + reconciler := &GrafanaStackReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("GrafanaStack"), + GrafanaCloudClient: gcClient, + GrafanaStackChangeEvents: grafanaStackChangeEvents, + OriginalStacks: map[string]struct{}{}, + } + c := make(chan time.Time) + wg := sync.WaitGroup{} + wg.Add(1) + go func() { + assert.NoError(t, reconciler.WatchGrafanaStacksChange(c)) + wg.Done() + }() + c <- time.Now() + close(c) + wg.Wait() + assert.EqualValues(t, 2, gcClient.ListStacksCalls) +} +func TestWatchGrafanaStacksChangeFailsHardWhenInitialListFails(t *testing.T) { + grafanaStackChangeEvents := make(chan event.GenericEvent, 10) + gcClient := &MockStackListerFunc{ + ListStacksFunc: func() (Stacks, error) { + return nil, errors.New("test-error") + }, + } + reconciler := &GrafanaStackReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("GrafanaStack"), + GrafanaCloudClient: gcClient, + GrafanaStackChangeEvents: grafanaStackChangeEvents, + OriginalStacks: map[string]struct{}{}, + } + c := make(chan time.Time) + close(c) + assert.Error(t, reconciler.WatchGrafanaStacksChange(c)) +} +func TestGrafanaStackReconcileReturnsAnErrorWhenClientListFails(t *testing.T) { + grafanaStackChangeEvents := make(chan event.GenericEvent) + gcClient := &MockStackListerFunc{ + ListStacksFunc: func() (Stacks, error) { + return nil, errors.New("test-error") + }, + } + reconciler := &GrafanaStackReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("GrafanaStack"), + GrafanaCloudClient: gcClient, + GrafanaStackChangeEvents: grafanaStackChangeEvents, + OriginalStacks: map[string]struct{}{}, + } + wg := sync.WaitGroup{} + wg.Add(1) + go func() { + for evt := range grafanaStackChangeEvents { + switch evt.Object.GetName() { + default: + assert.Fail(t, "should trigger a reconciliation of grafanacloud stack %s", evt.Object.GetName()) + } + } + wg.Done() + }() + assert.Error(t, reconciler.reconcile()) + assert.EqualValues(t, 1, gcClient.ListStacksCalls) + close(grafanaStackChangeEvents) + assert.Equal(t, map[string]struct{}{}, reconciler.OriginalStacks) + wg.Wait() +} +func TestGrafanaStackReconcileTriggersUpdateForAddedAndDeletedStacks(t *testing.T) { + grafanaStackChangeEvents := make(chan event.GenericEvent) + gcClient := &MockStackListerFunc{ + ListStacksFunc: func() (Stacks, error) { + return []Stack{ + {Slug: "adevintatenant", LogsInstanceID: 1234, LogsURL: "https://logs.grafanacloud.de"}, + {Slug: "adevintaruntime", LogsInstanceID: 9876, LogsURL: "https://logs.grafanacloud.es"}, + {Slug: "adevintanewtenants1", LogsInstanceID: 9878, LogsURL: "https://logs.grafanacloud.fr"}, + }, nil + }, + } + reconciler := &GrafanaStackReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("GrafanaStack"), + GrafanaCloudClient: gcClient, + GrafanaStackChangeEvents: grafanaStackChangeEvents, + OriginalStacks: map[string]struct{}{ + "adevintatenant": {}, + "adevintaothertenant": {}, + "adevintaruntime": {}, + }, + } + wg := sync.WaitGroup{} + wg.Add(1) + go func() { + for evt := range grafanaStackChangeEvents { + switch evt.Object.GetName() { + case "adevintanewtenants1": + case "adevintaothertenant": + default: + assert.Fail(t, "unexpected reconciliation of grafanacloud stack %s", evt.Object.GetName()) + } + } + wg.Done() + }() + require.NoError(t, reconciler.reconcile()) + assert.EqualValues(t, 1, gcClient.ListStacksCalls) + close(grafanaStackChangeEvents) + assert.Equal(t, map[string]struct{}{ + "adevintatenant": {}, + "adevintanewtenants1": {}, + "adevintaruntime": {}, + }, reconciler.OriginalStacks) + wg.Wait() +} +func TestGrafanaStackReconcileStackNoChangesDoesNotPopulateEvents(t *testing.T) { + grafanaStackChangeEvents := make(chan event.GenericEvent) + gcClient := &MockStackListerFunc{ + ListStacksFunc: func() (Stacks, error) { + return []Stack{ + {Slug: "adevintatenant", LogsInstanceID: 1234, LogsURL: "https://logs.grafanacloud.de"}, + {Slug: "adevintaruntime", LogsInstanceID: 9876, LogsURL: "https://logs.grafanacloud.es"}, + {Slug: "adevintaothertenant", LogsInstanceID: 9878, LogsURL: "https://logs.grafanacloud.fr"}, + }, nil + }, + } + reconciler := &GrafanaStackReconciler{ + Log: ctrl.Log.WithName("controllers").WithName("GrafanaStack"), + GrafanaCloudClient: gcClient, + GrafanaStackChangeEvents: grafanaStackChangeEvents, + OriginalStacks: map[string]struct{}{ + "adevintatenant": {}, + "adevintaothertenant": {}, + "adevintaruntime": {}, + }, + } + wg := sync.WaitGroup{} + wg.Add(1) + go func() { + for evt := range grafanaStackChangeEvents { + switch evt.Object.GetName() { + default: + assert.Fail(t, "unexpected reconciliation of grafanacloud stack %s", evt.Object.GetName()) + } + } + wg.Done() + }() + require.NoError(t, reconciler.reconcile()) + assert.EqualValues(t, 1, gcClient.ListStacksCalls) + close(grafanaStackChangeEvents) + assert.Equal(t, reconciler.OriginalStacks, map[string]struct{}{ + "adevintatenant": {}, + "adevintaothertenant": {}, + "adevintaruntime": {}, + }) + wg.Wait() +} diff --git a/pkg/grafanacloud/grafanacloud_cache.go b/pkg/grafanacloud/grafanacloud_cache.go new file mode 100644 index 0000000..07216df --- /dev/null +++ b/pkg/grafanacloud/grafanacloud_cache.go @@ -0,0 +1,128 @@ +package grafanacloud + +import ( + "fmt" + "sync" + "time" + + "github.com/go-logr/logr" +) + +type CacheEntry struct { + stack *Stack + timestamp int64 +} + +func (c CacheEntry) Value() *Stack { + return c.stack +} + +// IsStale checks if the entry is older than the amount passed for +// staleness in seconds. +func (c CacheEntry) IsStale(staleness int) bool { + return time.Now().Unix()-c.timestamp > int64(staleness) +} + +type StackCache struct { + sync.RWMutex + data map[string]CacheEntry + + // Last time we updated all entries + timestamp int64 +} + +func NewStackCache() StackCache { + data := make(map[string]CacheEntry) + return StackCache{data: data} +} + +type CachedClient struct { + Log logr.Logger + + client *Client + cache StackCache +} + +func NewCachedClient(logr logr.Logger, client *Client) *CachedClient { + return &CachedClient{ + Log: logr, + client: client, + cache: NewStackCache(), + } +} + +// GetStack returns a stack definition for the corresponding GrafanaCloud stack +func (c *CachedClient) GetStack(slug string) (*Stack, error) { + c.cache.RLock() + timestamp := c.cache.timestamp + c.cache.RUnlock() + + if timestamp == 0 { + // cache is empty, fill it with the data we have + if err := c.loadCache(600); err != nil { + return nil, fmt.Errorf("failed to update cache: %w", err) + } + } + + c.cache.RLock() + entry, ok := c.cache.data[slug] + c.cache.RUnlock() + + if ok && !entry.IsStale(300) { + // Recent cache entry + return entry.Value(), nil + } + + s, err := c.client.GetStack(slug) + if err != nil { + return nil, err + } + c.cache.Lock() + c.cache.data[slug] = CacheEntry{stack: s, timestamp: time.Now().Unix()} + c.cache.Unlock() + + return s, nil +} + +func (c *CachedClient) GetTracesConnection(stackSlug string) (int, string, error) { + return c.client.GetTracesConnection(stackSlug) +} + +func (c *CachedClient) ListStacks() (Stacks, error) { + if c.cache.timestamp == 0 || time.Now().Unix()-c.cache.timestamp > 600 { + if err := c.loadCache(600); err != nil { + return nil, fmt.Errorf("failed to update cache: %w", err) + } + } + + stacks := []Stack{} + for _, entry := range c.cache.data { + stacks = append(stacks, *entry.Value()) + } + + return stacks, nil +} + +func (c *CachedClient) loadCache(staleness int) error { + c.cache.Lock() + defer c.cache.Unlock() + + if time.Now().Unix()-c.cache.timestamp < int64(staleness) { + // cache was already updated by some other thread + // while we were waiting for the lock + return nil + } + + stacks, err := c.client.ListStacks() + if err != nil { + return err + } + + clear(c.cache.data) + for _, stack := range stacks { + c.cache.data[stack.Slug] = CacheEntry{stack: &stack, timestamp: time.Now().Unix()} + } + + c.cache.timestamp = time.Now().Unix() + return nil +} diff --git a/pkg/grafanacloud/grafanacloud_cache_test.go b/pkg/grafanacloud/grafanacloud_cache_test.go new file mode 100644 index 0000000..43dd85a --- /dev/null +++ b/pkg/grafanacloud/grafanacloud_cache_test.go @@ -0,0 +1,249 @@ +package grafanacloud + +import ( + "bytes" + "fmt" + "io" + "net/http" + "sync" + "testing" + "time" + + "github.com/go-logr/logr" + gcom "github.com/grafana/grafana-com-public-clients/go/gcom" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" +) + +const ( + bodyFound = `{"items":[{"hlInstanceId":123,"hmInstancePromId":456,"hmInstancePromUrl":"http://prom-prod.test","hlInstanceUrl":"http://logs-prod.test","id":789,"slug":"dummystack","url":"http://stack-prod.test"}]}` +) + +type httpTransportSpy struct { + Calls int + ResponseFunc func(*http.Request) (*http.Response, error) +} + +func (c *httpTransportSpy) RoundTrip(req *http.Request) (*http.Response, error) { + c.Calls++ + return c.ResponseFunc(req) +} + +func defaultResponseFunc(statusCode int, body string) func(req *http.Request) (*http.Response, error) { + return func(*http.Request) (*http.Response, error) { + resp := http.Response{ + StatusCode: statusCode, + Header: http.Header{ + "Content-Type": []string{"application/json"}, + "Content-Length": []string{fmt.Sprintf("%d", len(body))}, + }, + Body: io.NopCloser(bytes.NewReader([]byte(body))), + } + return &resp, nil + } +} + +func mockedGCOMClient(client *http.Client) *gcom.APIClient { + config := gcom.NewConfiguration() + config.Host = "grafana.com" + config.Scheme = "https" + config.HTTPClient = client + + return gcom.NewAPIClient(config) +} + +func TestGetStackUsesCache(t *testing.T) { + t.Run("when asking for the same stack twice, only one http call is made", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusOK, bodyFound), + } + + httpClient := &http.Client{ + Transport: transportSpy, + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(httpClient), + "test-org", + )) + + _, err := grafanaClient.GetStack("dummystack") + require.NoError(t, err) + + _, err = grafanaClient.GetStack("dummystack") + require.NoError(t, err) + + assert.Equal(t, 1, transportSpy.Calls) + }) + + t.Run("when asking for non-existing stacks, all trigger an http call", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusNotFound, `{"message":"Stack not found"}`), + } + + httpClient := &http.Client{ + Transport: transportSpy, + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(httpClient), + "test-org", + )) + + _, err := grafanaClient.GetStack("nonexistingstack") + require.Error(t, err) + + _, err = grafanaClient.GetStack("neitherdoesthisone") + require.Error(t, err) + + assert.Equal(t, 2, transportSpy.Calls) + }) +} + +func TestGetStackCacheStaleness(t *testing.T) { + t.Run("when the cache entry is not stale, we do not trigger any http call", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusOK, bodyFound), + } + + httpClient := &http.Client{ + Transport: transportSpy, + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(httpClient), + "test-org", + )) + + grafanaClient.cache.timestamp = time.Now().Unix() + + cacheContents := make(map[string]CacheEntry) + cacheContents["dummystack"] = CacheEntry{ + stack: &Stack{}, + timestamp: time.Now().Unix(), + } + + grafanaClient.cache.data = cacheContents + + _, err := grafanaClient.GetStack("dummystack") + + require.NoError(t, err) + + assert.Equal(t, 0, transportSpy.Calls) + }) + t.Run("when the cache entry is stale, we trigger and http call", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusOK, bodyFound), + } + + httpClient := &http.Client{ + Transport: transportSpy, + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(httpClient), + "test-org", + )) + + cacheContents := make(map[string]CacheEntry) + cacheContents["dummystack"] = CacheEntry{ + stack: &Stack{}, + timestamp: time.Now().Add(-time.Hour).Unix(), + } + + grafanaClient.cache.data = cacheContents + + _, err := grafanaClient.GetStack("dummystack") + require.NoError(t, err) + + assert.Equal(t, 1, transportSpy.Calls) + }) +} + +func TestConcurrentGetStackCache(t *testing.T) { + t.Run("concurrent access to the cache is safe", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusOK, bodyFound), + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(&http.Client{ + Transport: transportSpy, + }), + "test-org", + )) + + var wg sync.WaitGroup + + // We simulate concurrent access to the cache + for range 100 { + wg.Add(1) + go func() { + _, err := grafanaClient.GetStack("dummystack") + require.NoError(t, err) + wg.Done() + }() + } + wg.Wait() + + assert.Equal(t, 1, transportSpy.Calls) + }) +} + +func TestListStackCache(t *testing.T) { + t.Run("Call ListStacks when the cache is empty. Makes one call to the Grafana API, to populate the cache, then uses it to respond the rest of the calls", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusOK, bodyFound), + } + + httpClient := &http.Client{ + Transport: transportSpy, + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(httpClient), + "test-org", + )) + + _, err := grafanaClient.ListStacks() + require.NoError(t, err) + + _, err = grafanaClient.ListStacks() + require.NoError(t, err) + + assert.Equal(t, 1, transportSpy.Calls) + }) + + t.Run("Call ListStacks when the cache is stale. Makes one call to the Grafana API, to populate the cache, then uses it to respond the rest of the calls", func(t *testing.T) { + transportSpy := &httpTransportSpy{ + ResponseFunc: defaultResponseFunc(http.StatusOK, bodyFound), + } + + httpClient := &http.Client{ + Transport: transportSpy, + } + + grafanaClient := NewCachedClient(logr.Logger{}, NewClient( + logr.Logger{}, + mockedGCOMClient(httpClient), + "test-org", + )) + + _, err := grafanaClient.ListStacks() + require.NoError(t, err) + + grafanaClient.cache.timestamp = 0 + + _, err = grafanaClient.ListStacks() + require.NoError(t, err) + + assert.Equal(t, 2, transportSpy.Calls) + }) + +} diff --git a/pkg/grafanacloud/grafanacloud_public.go b/pkg/grafanacloud/grafanacloud_public.go new file mode 100644 index 0000000..1461eae --- /dev/null +++ b/pkg/grafanacloud/grafanacloud_public.go @@ -0,0 +1,131 @@ +package grafanacloud + +import ( + "context" + "fmt" + "net/http" + "strconv" + + "github.com/go-logr/logr" + gcom "github.com/grafana/grafana-com-public-clients/go/gcom" +) + +var Config *config = NewConfig("adevinta.com") + +type config struct { + // Destination GrafanaCloud stack selection + stackNameAnnotationKey string + // Namespace feature toggles + logsLabelKey string +} + +func NewConfig(domain string) *config { + return &config{ + stackNameAnnotationKey: "grafanacloud." + domain + "/stack-name", + logsLabelKey: "grafanacloud." + domain + "/logs", + } +} + +// Stack contains all the relevant details of a GrafanaCloud stack +type Stack struct { + LogsInstanceID int `json:"hlInstanceId"` + MetricsInstanceID int `json:"hmInstancePromId"` + PromURL string `json:"hmInstancePromUrl"` + LogsURL string `json:"hlInstanceUrl"` + StackID int `json:"id"` + Slug string `json:"slug" yaml:"slug"` + URL string `json:"url" yaml:"url"` +} + +type Stacks []Stack + +type Client struct { + GComClient *gcom.APIClient + Log logr.Logger + OrgSlug string +} + +func NewClient(logr logr.Logger, gcomClient *gcom.APIClient, org string) *Client { + return &Client{ + GComClient: gcomClient, + Log: logr, + OrgSlug: org, + } +} + +// GetStack returns a stack definition for the corresponding GrafanaCloud stack +func (c *Client) GetStack(slug string) (*Stack, error) { + resp, httpResp, err := c.GComClient.InstancesAPI.GetInstances(context.Background()).OrgSlug(c.OrgSlug).Slug(slug).Execute() + if err != nil { + return nil, fmt.Errorf("failed to get stack %s: %s", slug, err.Error()) + } + if httpResp.StatusCode != http.StatusOK { + return nil, fmt.Errorf("unexpected return code") + } + + if len(resp.GetItems()) == 0 { + err := fmt.Errorf("stack not found: %s", slug) + return nil, err + } + + stack := resp.GetItems()[0] + s := &Stack{ + LogsInstanceID: int(stack.HlInstanceId), + MetricsInstanceID: int(stack.HmInstancePromId), + PromURL: stack.HmInstancePromUrl, + LogsURL: stack.HlInstanceUrl, + StackID: int(stack.Id), + Slug: stack.Slug, + URL: stack.Url, + } + + return s, nil +} + +func (c *Client) GetTracesConnection(stackSlug string) (int, string, error) { + stack, err := c.GetStack(stackSlug) + if err != nil { + return -1, "", err + } + resp, httpResp, err := c.GComClient.InstancesAPI.GetConnections(context.Background(), strconv.Itoa(stack.StackID)).Execute() + if err != nil { + return -1, "", err + } + if httpResp.StatusCode != http.StatusOK { + return -1, "Cannot retrieve the OTLP connection: ", fmt.Errorf("unexpected return code") + } + otlpURL := resp.OtlpHttpUrl + if !otlpURL.IsSet() || otlpURL.Get() == nil { + return -1, "", fmt.Errorf("OTLP URL is not set") + } + return stack.StackID, *otlpURL.Get(), nil +} + +func (c *Client) ListStacks() (Stacks, error) { + resp, httpResp, err := c.GComClient.InstancesAPI.GetInstances(context.Background()).OrgSlug(c.OrgSlug).Execute() + if err != nil { + return nil, fmt.Errorf("failed to get stacks: %s", err.Error()) + } + + if httpResp.StatusCode != http.StatusOK { + return nil, fmt.Errorf("unexpected return code") + } + + stacks := []Stack{} + for _, stack := range resp.Items { + stacks = append( + stacks, + Stack{ + LogsInstanceID: int(stack.HlInstanceId), + MetricsInstanceID: int(stack.HmInstancePromId), + PromURL: stack.HmInstancePromUrl, + LogsURL: stack.HlInstanceUrl, + StackID: int(stack.Id), + Slug: stack.Slug, + URL: stack.Url, + }, + ) + } + + return stacks, nil +} diff --git a/pkg/grafanacloud/grafanacloud_public_test.go b/pkg/grafanacloud/grafanacloud_public_test.go new file mode 100644 index 0000000..a72d0d2 --- /dev/null +++ b/pkg/grafanacloud/grafanacloud_public_test.go @@ -0,0 +1,79 @@ +package grafanacloud_test + +import ( + "os" + "testing" + + "github.com/adevinta/observability-operator/pkg/grafanacloud" + "github.com/go-logr/logr" + "github.com/grafana/grafana-com-public-clients/go/gcom" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" +) + +func defaultGCOMClient(token string) *gcom.APIClient { + config := gcom.NewConfiguration() + config.AddDefaultHeader("Authorization", "Bearer "+token) + config.Host = "grafana.com" + config.Scheme = "https" + + return gcom.NewAPIClient(config) +} + +func TestGetStack(t *testing.T) { + log := logr.Logger{} + + token, ok := os.LookupEnv("GRAFANA_CLOUD_TOKEN") + if !ok { + t.Skip("Missing GRAFANA_CLOUD_TOKEN") + } + + gcomClient := defaultGCOMClient(token) + + client := grafanacloud.NewClient(log, gcomClient, "adevinta") + + stack, err := client.GetStack("adevintatest") + require.NoError(t, err) + assert.Equal(t, 150026, stack.StackID) + assert.Equal(t, "adevintatest", stack.Slug) + assert.Equal(t, "https://adevintatest.grafana.net", stack.URL) + assert.Equal(t, "https://logs-prod-eu-west-0.grafana.net", stack.LogsURL) + assert.Equal(t, 8302, stack.LogsInstanceID) + assert.Equal(t, "https://prometheus-prod-01-eu-west-0.grafana.net", stack.PromURL) + assert.Equal(t, 18630, stack.MetricsInstanceID) +} + +func TestListStacks(t *testing.T) { + log := logr.Logger{} + + token, ok := os.LookupEnv("GRAFANA_CLOUD_TOKEN") + if !ok { + t.Skip("Missing GRAFANA_CLOUD_TOKEN") + } + + gcomClient := defaultGCOMClient(token) + + client := grafanacloud.NewClient(log, gcomClient, "adevinta") + + stacks, err := client.ListStacks() + require.NoError(t, err) + assert.Greater(t, len(stacks), 0) +} + +func TestGetTracesConnection(t *testing.T) { + log := logr.Logger{} + + token, ok := os.LookupEnv("GRAFANA_CLOUD_TOKEN") + if !ok { + t.Skip("Missing GRAFANA_CLOUD_TOKEN") + } + + gcomClient := defaultGCOMClient(token) + + client := grafanacloud.NewClient(log, gcomClient, "adevinta") + + num, url, err := client.GetTracesConnection("adevintatest") + require.NoError(t, err) + assert.Equal(t, url, "https://otlp-gateway-prod-eu-west-0.grafana.net") + assert.Equal(t, num, 150026) +} diff --git a/pkg/grafanacloud/loki_config.go b/pkg/grafanacloud/loki_config.go new file mode 100644 index 0000000..54105ec --- /dev/null +++ b/pkg/grafanacloud/loki_config.go @@ -0,0 +1,202 @@ +package grafanacloud + +import ( + "bytes" + "context" + _ "embed" + "errors" + "slices" + "strings" + "text/template" + + "github.com/go-logr/logr" + corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/client-go/util/workqueue" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +//go:embed fluentd_template.conf +var lokiFluentDTemplate string + +type GrafanaCloudStackLister interface { + ListStacks() (Stacks, error) +} + +type GrafanaCloudConfigUpdater struct { + client.Client + Log logr.Logger + GrafanaCloudClient GrafanaCloudStackLister + ClusterName string + ClusterRegion string + RateLimiter RateLimitedFunc + + ConfigMapNamespace string + ConfigMapName string + ConfigMapLokiKey string +} + +type RateLimitedFunc struct { + Do func(context.Context) + Queue workqueue.TypedRateLimitingInterface[string] +} + +func (r *RateLimitedFunc) Start() { + for { + item, stop := r.Queue.Get() + if stop { + return + } + r.Do(context.Background()) + r.Queue.Done(item) + } +} + +func (r *RateLimitedFunc) EnsureDone() { + switch r.Queue.Len() { + case 0: + r.Queue.Add("call") + case 1: + r.Queue.AddRateLimited("call") + } +} + +func (r *GrafanaCloudConfigUpdater) Start(queue workqueue.TypedRateLimitingInterface[string]) { + r.RateLimiter = RateLimitedFunc{ + Do: r.createFluentDConfigmap, + Queue: queue, + } + r.RateLimiter.Start() +} + +func (r *GrafanaCloudConfigUpdater) InjectFluentdLokiConfiguration(ctx context.Context) error { + if r.ConfigMapName == "" && r.ConfigMapNamespace == "" && r.ConfigMapLokiKey == "" { + return nil + } + if r.ConfigMapName == "" { + return errors.New("missing loki configmap name") + } + if r.ConfigMapNamespace == "" { + return errors.New("missing loki configmap namespace") + } + if r.ConfigMapLokiKey == "" { + return errors.New("missing loki configmap loki key") + } + r.RateLimiter.EnsureDone() + return nil +} + +func (r *GrafanaCloudConfigUpdater) createFluentDConfigmap(ctx context.Context) { + namespaces := corev1.NamespaceList{} + err := r.Client.List(ctx, &namespaces) + if err != nil { + r.Log.Error(err, "failed to list namespaces") + return + } + + stacks, err := r.GrafanaCloudClient.ListStacks() + if err != nil { + r.Log.Error(err, "failed to list grafanacloud stacks") + return + } + if len(stacks) == 0 { + r.Log.Info("Grafana API does not return any stack, skipping loki configuration injection") + return + } + r.Log.Info("namespaces and stacks", "namespaceCount", len(namespaces.Items), "stackCount", len(stacks)) + + lokiConfigs := map[string][]lokiCredentials{} + for _, namespace := range namespaces.Items { + if value, ok := namespace.Labels[Config.logsLabelKey]; ok && value == "disabled" { + // This namespace does not want logs, move to next one + r.Log.WithValues("namespace", namespace.GetName()).Info("namespace disabled log routing, skipping") + continue + } + + configuredStacks, ok := namespace.Annotations[Config.stackNameAnnotationKey] + if !ok { + r.Log.WithValues("namespace", namespace.GetName()).Info("namespace has no configured destination stack, skipping") + continue + } + + var stackNames []string + for _, part := range strings.Split(configuredStacks, ",") { + stackNames = append(stackNames, strings.TrimSpace(part)) + } + + for _, stack := range stacks { + if slices.IndexFunc(stackNames, func(name string) bool { return name == stack.Slug }) == -1 { + // stack is not present in the list of configured stacks for this namespace + continue + } + if lokiConfigs[namespace.Name] != nil { + // There's more than one destination stack for this namespace + lokiConfigs[namespace.Name] = append( + lokiConfigs[namespace.Name], + lokiCredentials{ + URL: stack.LogsURL, + UserID: stack.LogsInstanceID, + }, + ) + } else { + // This is the first destination stack we have found for this namespace + lokiConfigs[namespace.Name] = []lokiCredentials{ + { + URL: stack.LogsURL, + UserID: stack.LogsInstanceID, + }, + } + } + } + } + r.Log.Info("found loki configurations", "namespaceCount", len(lokiConfigs)) + + tpl := template.New("loki-config") + tpl, err = tpl.Parse(lokiFluentDTemplate) + if err != nil { + r.Log.Error(err, "failed to generate loki configuration") + return + } + b := bytes.Buffer{} + if err := tpl.Execute(&b, lokiOptions{ + Cluster: clusterDetails{ + Name: r.ClusterName, + Region: r.ClusterRegion, + }, + Stacks: lokiConfigs, + }); err != nil { + r.Log.Error(err, "failed to generate loki configuration") + return + } + cm := corev1.ConfigMap{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: r.ConfigMapNamespace, + Name: r.ConfigMapName, + }, + } + if _, err := ctrl.CreateOrUpdate(ctx, r.Client, &cm, func() error { + cm.Data = map[string]string{ + r.ConfigMapLokiKey: b.String(), + } + return nil + }); err != nil { + r.Log.Error(err, "failed to create loki configuration configmap") + return + } + r.Log.Info("injected loki configuration", "namespaceCount", len(lokiConfigs)) +} + +type lokiOptions struct { + Cluster clusterDetails + Stacks map[string][]lokiCredentials +} + +type clusterDetails struct { + Name, Region string +} + +type lokiCredentials struct { + URL string + UserID int +} diff --git a/pkg/grafanacloud/loki_config_test.go b/pkg/grafanacloud/loki_config_test.go new file mode 100644 index 0000000..f6f3cc1 --- /dev/null +++ b/pkg/grafanacloud/loki_config_test.go @@ -0,0 +1,1017 @@ +package grafanacloud + +import ( + "bytes" + "context" + "errors" + "strings" + "testing" + "text/template" + "time" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + corev1 "k8s.io/api/core/v1" + k8serrors "k8s.io/apimachinery/pkg/api/errors" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/runtime/schema" + "k8s.io/client-go/util/workqueue" + ctrl "sigs.k8s.io/controller-runtime" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/client/fake" +) + +type GrafanaCloudClientFunc struct { + ListStacksCalls int + ListStacksFunc func() (Stacks, error) +} + +func (cg *GrafanaCloudClientFunc) ListStacks() (Stacks, error) { + cg.ListStacksCalls++ + if cg.ListStacksFunc != nil { + return cg.ListStacksFunc() + } + return nil, errors.New("ListStacks not implemented") +} + +// Allow to reduce the number functions to write by composing with the interface. +type testK8sLister struct { + client.Client + ListCalls int + ListFunc func(ctx context.Context, list client.ObjectList, opts ...client.ListOption) error + GetCalls int + GetFunc func(ctx context.Context, key client.ObjectKey, obj client.Object, opts ...client.GetOption) error + UpdateCalls int + UpdateFunc func(ctx context.Context, obj client.Object, opts ...client.UpdateOption) error + CreateCalls int + CreateFunc func(ctx context.Context, obj client.Object, opts ...client.CreateOption) error +} + +func (t *testK8sLister) List(ctx context.Context, list client.ObjectList, opts ...client.ListOption) error { + t.ListCalls++ + if t.ListFunc != nil { + return t.ListFunc(ctx, list, opts...) + } + if t.Client != nil { + return t.Client.List(ctx, list, opts...) + } + return errors.New("List not implemented") +} + +func (t *testK8sLister) Get(ctx context.Context, key client.ObjectKey, obj client.Object, opts ...client.GetOption) error { + t.GetCalls++ + if t.GetFunc != nil { + return t.GetFunc(ctx, key, obj, opts...) + } + if t.Client != nil { + return t.Client.Get(ctx, key, obj, opts...) + } + return errors.New("Get not implemented") +} + +func (t *testK8sLister) Update(ctx context.Context, obj client.Object, opts ...client.UpdateOption) error { + t.UpdateCalls++ + if t.UpdateFunc != nil { + return t.UpdateFunc(ctx, obj, opts...) + } + if t.Client != nil { + return t.Client.Update(ctx, obj, opts...) + } + return errors.New("Update not implemented") +} + +func (t *testK8sLister) Create(ctx context.Context, obj client.Object, opts ...client.CreateOption) error { + t.CreateCalls++ + if t.CreateFunc != nil { + return t.CreateFunc(ctx, obj, opts...) + } + if t.Client != nil { + return t.Client.Create(ctx, obj, opts...) + } + return errors.New("Createnot implemented") +} + +func TestFluentdConfigTemplateCorrectness(t *testing.T) { + // This test verifies that changes to the template result in valid, executable templates + tpl := template.New("fluentd-config") + _, err := tpl.Parse(lokiFluentDTemplate) + require.NoError(t, err, "couldn't parse Fluentd configuration template successfully") + + t.Run("when there are no namespaces, fluentd config file generated is syntactically correct", func(t *testing.T) { + config := lokiOptions{ + Cluster: clusterDetails{ + Name: "my-cluster", + Region: "eu-fake-1", + }, + Stacks: map[string][]lokiCredentials{}, + } + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + require.NoError(t, err, "couldn't execute Fluentd configuration template successfully") + }) + t.Run("when a namespace has a single stack, fluentd config file generated is syntactically correct", func(t *testing.T) { + namespace := "my-namespace" + config := lokiOptions{ + Cluster: clusterDetails{ + Name: "my-cluster", + Region: "eu-fake-1", + }, + Stacks: map[string][]lokiCredentials{ + namespace: { + { + URL: "http://grafana.net/loki", + UserID: 123, + }, + }, + }, + } + + expectedFluentdconf := ` + @type record_modifier + @id inject_loki_user_environment_dev + + environment "dev" + + + + + @type record_modifier + @id inject_loki_user_environment_pre + + environment "pre" + + + + + @type record_modifier + @id inject_loki_user_environment_pro + + environment "pro" + + + + @type copy + + @type loki + @id loki_loki_my-namespace_access_log_0 + url "http://grafana.net/loki" + username "123" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 1m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + + + @type copy + + @type loki + @id loki_loki_my-namespace_0 + url "http://grafana.net/loki" + username "123" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 64m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + +` + + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + require.NoError(t, err, "couldn't execute Fluentd configuration template successfully") + require.Equal(t, expectedFluentdconf, buffer.String()) + }) + t.Run("when a namespace has multiple stacks, fluentd config file generated is syntactically correct", func(t *testing.T) { + config := lokiOptions{ + Cluster: clusterDetails{ + Name: "my-cluster", + Region: "eu-fake-1", + }, + Stacks: map[string][]lokiCredentials{ + "my-namespace": { + { + URL: "http://grafana.net/loki", + UserID: 123, + }, + { + URL: "http://grafana.net/loki", + UserID: 456, + }, + }, + }, + } + + expectedFluentdconf := ` + @type record_modifier + @id inject_loki_user_environment_dev + + environment "dev" + + + + + @type record_modifier + @id inject_loki_user_environment_pre + + environment "pre" + + + + + @type record_modifier + @id inject_loki_user_environment_pro + + environment "pro" + + + + @type copy + + @type loki + @id loki_loki_my-namespace_access_log_0 + url "http://grafana.net/loki" + username "123" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 1m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + @type loki + @id loki_loki_my-namespace_access_log_1 + url "http://grafana.net/loki" + username "456" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 1m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + + + @type copy + + @type loki + @id loki_loki_my-namespace_0 + url "http://grafana.net/loki" + username "123" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 64m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + @type loki + @id loki_loki_my-namespace_1 + url "http://grafana.net/loki" + username "456" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 64m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + +` + + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + require.NoError(t, err, "couldn't execute Fluentd configuration template successfully") + require.Equal(t, expectedFluentdconf, buffer.String()) + }) + t.Run("when we have several namespaces, fluentd config file generated is syntactically correct", func(t *testing.T) { + config := lokiOptions{ + Cluster: clusterDetails{ + Name: "my-cluster", + Region: "eu-fake-1", + }, + Stacks: map[string][]lokiCredentials{ + "my-namespace": { + { + URL: "http://grafana.net/loki", + UserID: 123, + }, + }, + "other-namespace": []lokiCredentials{ + { + URL: "http://grafana.net/loki", + UserID: 456, + }, + }, + }, + } + + expectedFluentdconf := ` + @type record_modifier + @id inject_loki_user_environment_dev + + environment "dev" + + + + + @type record_modifier + @id inject_loki_user_environment_pre + + environment "pre" + + + + + @type record_modifier + @id inject_loki_user_environment_pro + + environment "pro" + + + + @type copy + + @type loki + @id loki_loki_my-namespace_access_log_0 + url "http://grafana.net/loki" + username "123" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 1m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + + + @type copy + + @type loki + @id loki_loki_my-namespace_0 + url "http://grafana.net/loki" + username "123" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 64m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + + @type copy + + @type loki + @id loki_loki_other-namespace_access_log_0 + url "http://grafana.net/loki" + username "456" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 1m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + + + + @type copy + + @type loki + @id loki_loki_other-namespace_0 + url "http://grafana.net/loki" + username "456" + password "#{ENV['LOKI_PASSWORD']}" + buffer_chunk_limit 64m + + flush_interval 60s # default + flush_at_shutdown true # default + retry_timeout 60s # instead of the default 72h + retry_max_times 5 # instead of the default 17 (i.e. 65536 seconds, or 1092 minutes or 18 hours) + retry_max_interval 10 + flush_thread_count 1 # default + overflow_action drop_oldest_chunk + disable_chunk_backup true + + # this instructs loki to send each "fluentd record" as JSON + # i.e. '{"log": "${lineLoggedByUser}", "time": "2021-01-01T00:00", "kubernetes": {"namespace": {"name": "${namespaceName}"}}}' + # removing this line transforms the record in kubernetes={"namespace": {"name": "${namespaceName}"}}} time=2021-01-01T00:00 log=${lineLoggedByUser} + # that is unparsable on the loki query side + line_format json + extra_labels {"cluster": "my-cluster","region":"eu-fake-1"} + remove_keys _sumo_metadata,source,category,host + + + +` + + var buffer bytes.Buffer + err = tpl.Execute(&buffer, config) + require.NoError(t, err, "couldn't execute Fluentd configuration template successfully") + require.Equal(t, expectedFluentdconf, buffer.String()) + }) +} + +func TestRateLimiterFunc(t *testing.T) { + calls := 0 + f := RateLimitedFunc{ + Do: func(context.Context) { + calls += 1 + }, + Queue: workqueue.NewTypedRateLimitingQueue(workqueue.NewTypedItemExponentialFailureRateLimiter[string](1*time.Millisecond, 100*time.Millisecond)), + } + go f.Start() + for i := 0; i < 100; i++ { + f.EnsureDone() + } + time.Sleep(400 * time.Millisecond) + assert.InDelta(t, 3, calls, 4) +} + +func TestGrafanaCloudCreateLokiConfigMaps(t *testing.T) { + k8sClient := testK8sLister{ + Client: fake.NewClientBuilder().WithObjects( + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "tenant-dev", Labels: map[string]string{Config.logsLabelKey: "disabled"}}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "other-tenant-pro", Annotations: map[string]string{Config.stackNameAnnotationKey: "adevintaothertenant"}}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "another-tenant-pro", Labels: map[string]string{Config.logsLabelKey: "enabled"}, Annotations: map[string]string{Config.stackNameAnnotationKey: "adevintaanothertenant"}}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "tenant-without-env-in-name", Annotations: map[string]string{Config.stackNameAnnotationKey: "adevintatenant"}}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "tenant-not-in-gc-pro"}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{ + Name: "system", + Annotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }, + }}, + ).Build(), + CreateFunc: func(ctx context.Context, obj client.Object, opts ...client.CreateOption) error { + if cm, ok := obj.(*corev1.ConfigMap); ok { + assert.Equal(t, "test-platform-services", cm.Namespace) + assert.Equal(t, "test-fluentd-config", cm.Name) + require.Contains(t, cm.Data, "loki") + assert.NotContains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-dev_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_other-tenant-pro_0`, + ` url "https://logs.grafanacloud.com"`, + ` username "5678"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_another-tenant-pro_0`, + ` url "https://logs.grafanacloud.com"`, + ` username "6789"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-without-env-in-name_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.NotContains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-dev_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_system_0`, + ` url "https://logs.grafanacloud.es"`, + ` username "9876"`, + }, + "\n", + )) + // ingress access logs + assert.NotContains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-dev_access_log_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_other-tenant-pro_access_log_0`, + ` url "https://logs.grafanacloud.com"`, + ` username "5678"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_another-tenant-pro_access_log_0`, + ` url "https://logs.grafanacloud.com"`, + ` username "6789"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-without-env-in-name_access_log_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.NotContains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-dev_access_log_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_system_access_log_0`, + ` url "https://logs.grafanacloud.es"`, + ` username "9876"`, + }, + "\n", + )) + assert.NotContains(t, cm.Data["loki"], "tenant-not-in-gc-pro") + return nil + } + t.Errorf("unexpected object create %T", obj) + return errors.New("unexpected object") + }, + } + gcClient := &GrafanaCloudClientFunc{ + ListStacksFunc: func() (Stacks, error) { + return []Stack{ + {Slug: "adevintatenant", LogsInstanceID: 1234, LogsURL: "https://logs.grafanacloud.de"}, + {Slug: "adevintaothertenant", LogsInstanceID: 5678, LogsURL: "https://logs.grafanacloud.com"}, + {Slug: "adevintaanothertenant", LogsInstanceID: 6789, LogsURL: "https://logs.grafanacloud.com"}, + {Slug: "adevintaruntime", LogsInstanceID: 9876, LogsURL: "https://logs.grafanacloud.es"}, + }, nil + }, + } + updater := &GrafanaCloudConfigUpdater{ + Client: &k8sClient, + Log: ctrl.Log.WithName("controllers").WithName("Namespace"), + GrafanaCloudClient: gcClient, + ClusterName: "test-cluster", + ClusterRegion: "adv-bcn-1", + ConfigMapNamespace: "test-platform-services", + ConfigMapName: "test-fluentd-config", + ConfigMapLokiKey: "loki", + } + go updater.Start(workqueue.NewTypedRateLimitingQueue(workqueue.NewTypedItemExponentialFailureRateLimiter[string](10*time.Microsecond, 1*time.Millisecond))) + time.Sleep(500 * time.Microsecond) + for i := 0; i < 100; i++ { + err := updater.InjectFluentdLokiConfiguration(context.Background()) + require.NoError(t, err) + } + time.Sleep(10 * time.Millisecond) + assert.LessOrEqual(t, k8sClient.GetCalls, 20) + assert.LessOrEqual(t, k8sClient.CreateCalls, 20) + assert.GreaterOrEqual(t, k8sClient.GetCalls, 1) + assert.GreaterOrEqual(t, k8sClient.CreateCalls, 1) +} + +func TestGrafanaCloudCreateLokiConfigMapsForMultipleStacks(t *testing.T) { + k8sClient := testK8sLister{ + Client: fake.NewClientBuilder().WithObjects( + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "tenant-dev", Annotations: map[string]string{Config.stackNameAnnotationKey: "adevintatenant"}}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{ + Name: "other-tenant-pro", + Annotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintastack1,adevintastack2", + }, + }}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "tenant-not-in-gc-pro"}}, + &corev1.Namespace{ObjectMeta: metav1.ObjectMeta{ + Name: "system", + Annotations: map[string]string{ + Config.stackNameAnnotationKey: "adevintaruntime", + }, + }}, + ).Build(), + CreateFunc: func(ctx context.Context, obj client.Object, opts ...client.CreateOption) error { + if cm, ok := obj.(*corev1.ConfigMap); ok { + assert.Equal(t, "test-platform-services", cm.Namespace) + assert.Equal(t, "test-fluentd-config", cm.Name) + require.Contains(t, cm.Data, "loki") + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-dev_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_other-tenant-pro_0`, + ` url "https://logs.grafanacloud.com"`, + ` username "5566"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ` `, + ` @type loki`, + ` @id loki_loki_other-tenant-pro_1`, + ` url "https://logs.grafanacloud.fr"`, + ` username "6655"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_system_0`, + ` url "https://logs.grafanacloud.es"`, + ` username "9876"`, + }, + "\n", + )) + // ingress access logs + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_tenant-dev_access_log_0`, + ` url "https://logs.grafanacloud.de"`, + ` username "1234"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_other-tenant-pro_access_log_0`, + ` url "https://logs.grafanacloud.com"`, + ` username "5566"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ` `, + ` @type loki`, + ` @id loki_loki_other-tenant-pro_access_log_1`, + ` url "https://logs.grafanacloud.fr"`, + ` username "6655"`, + }, + "\n", + )) + assert.Contains(t, cm.Data["loki"], strings.Join([]string{ + ``, + ` @type copy`, + ` `, + ` @type loki`, + ` @id loki_loki_system_access_log_0`, + ` url "https://logs.grafanacloud.es"`, + ` username "9876"`, + }, + "\n", + )) + assert.NotContains(t, cm.Data["loki"], "tenant-not-in-gc-pro") + return nil + } + t.Errorf("unexpected object create %T", obj) + return errors.New("unexpected object") + }, + } + gcClient := &GrafanaCloudClientFunc{ + ListStacksFunc: func() (Stacks, error) { + return []Stack{ + {Slug: "adevintatenant", LogsInstanceID: 1234, LogsURL: "https://logs.grafanacloud.de"}, + {Slug: "adevintaothertenant", LogsInstanceID: 5678, LogsURL: "https://logs.grafanacloud.com"}, + {Slug: "adevintaruntime", LogsInstanceID: 9876, LogsURL: "https://logs.grafanacloud.es"}, + {Slug: "adevintastack1", LogsInstanceID: 5566, LogsURL: "https://logs.grafanacloud.com"}, + {Slug: "adevintastack2", LogsInstanceID: 6655, LogsURL: "https://logs.grafanacloud.fr"}, + }, nil + }, + } + updater := &GrafanaCloudConfigUpdater{ + Client: &k8sClient, + Log: ctrl.Log.WithName("controllers").WithName("Namespace"), + GrafanaCloudClient: gcClient, + ClusterName: "test-cluster", + ClusterRegion: "adv-bcn-1", + ConfigMapNamespace: "test-platform-services", + ConfigMapName: "test-fluentd-config", + ConfigMapLokiKey: "loki", + } + go updater.Start(workqueue.NewTypedRateLimitingQueue(workqueue.NewTypedItemExponentialFailureRateLimiter[string](10*time.Microsecond, 1*time.Millisecond))) + time.Sleep(500 * time.Microsecond) + for i := 0; i < 100; i++ { + err := updater.InjectFluentdLokiConfiguration(context.Background()) + require.NoError(t, err) + } + time.Sleep(10 * time.Millisecond) + assert.LessOrEqual(t, k8sClient.GetCalls, 20) + assert.LessOrEqual(t, k8sClient.CreateCalls, 20) + assert.GreaterOrEqual(t, k8sClient.GetCalls, 1) + assert.GreaterOrEqual(t, k8sClient.CreateCalls, 1) +} + +func TestGrafanaCloudEmptyListDoesNotUpdateConfig(t *testing.T) { + k8sClient := testK8sLister{ + GetFunc: func(ctx context.Context, key client.ObjectKey, obj client.Object, opts ...client.GetOption) error { + return k8serrors.NewNotFound(schema.GroupResource{}, "resource") + }, + ListFunc: func(ctx context.Context, list client.ObjectList, opts ...client.ListOption) error { + if namespaces, ok := list.(*corev1.NamespaceList); ok { + namespaces.Items = []corev1.Namespace{ + {ObjectMeta: metav1.ObjectMeta{Name: "tenant-dev"}}, + } + return nil + } + t.Errorf("unexpected list object %T", list) + return errors.New("unexpected object") + }, + CreateFunc: func(ctx context.Context, obj client.Object, opts ...client.CreateOption) error { + if cm, ok := obj.(*corev1.ConfigMap); ok { + assert.Equal(t, "test-platform-services", cm.Namespace) + assert.Equal(t, "test-fluentd-config", cm.Name) + return nil + } + t.Errorf("unexpected object create %T", obj) + return errors.New("unexpected object") + }, + } + gcClient := &GrafanaCloudClientFunc{ + ListStacksFunc: func() (Stacks, error) { + return []Stack{}, nil + }, + } + updater := &GrafanaCloudConfigUpdater{ + Client: &k8sClient, + Log: ctrl.Log.WithName("controllers").WithName("Namespace"), + GrafanaCloudClient: gcClient, + ClusterName: "test-cluster", + ClusterRegion: "adv-bcn-1", + ConfigMapNamespace: "test-platform-services", + ConfigMapName: "test-fluentd-config", + ConfigMapLokiKey: "loki", + } + go updater.Start(workqueue.NewTypedRateLimitingQueue(workqueue.NewTypedItemExponentialFailureRateLimiter[string](10*time.Microsecond, 1*time.Millisecond))) + time.Sleep(500 * time.Microsecond) + for i := 0; i < 100; i++ { + err := updater.InjectFluentdLokiConfiguration(context.Background()) + require.NoError(t, err) + } + time.Sleep(10 * time.Millisecond) + assert.Equal(t, 0, k8sClient.CreateCalls) +} diff --git a/pkg/test_helpers/http.go b/pkg/test_helpers/http.go new file mode 100644 index 0000000..30dac4d --- /dev/null +++ b/pkg/test_helpers/http.go @@ -0,0 +1,32 @@ +package test_helpers + +import ( + "bytes" + "io" + "net/http" +) + +type RoundTripFunc func(req *http.Request) *http.Response + +func (f RoundTripFunc) RoundTrip(req *http.Request) (*http.Response, error) { + return f(req), nil +} +func newTestClient(fn RoundTripFunc) *http.Client { + return &http.Client{ + Transport: RoundTripFunc(fn), + } +} + +func NewHttpMockWithResponse(response string, statusCode int) *http.Client { + return newTestClient(func(req *http.Request) *http.Response { + return &http.Response{ + StatusCode: statusCode, + Body: io.NopCloser(bytes.NewBufferString(response)), + Header: make(http.Header), + } + }) +} + +func NewHttpMockWithFunc(fn func(req *http.Request) *http.Response) *http.Client { + return newTestClient(fn) +} diff --git a/pkg/test_helpers/test_helpers.go b/pkg/test_helpers/test_helpers.go new file mode 100644 index 0000000..245bf14 --- /dev/null +++ b/pkg/test_helpers/test_helpers.go @@ -0,0 +1,26 @@ +package test_helpers + +import ( + "testing" + + "k8s.io/apimachinery/pkg/runtime" + "k8s.io/kubectl/pkg/scheme" + "sigs.k8s.io/controller-runtime/pkg/client" + "sigs.k8s.io/controller-runtime/pkg/client/fake" + + monitoringv1 "github.com/prometheus-operator/prometheus-operator/pkg/apis/monitoring/v1" + "github.com/stretchr/testify/require" + vpav1 "k8s.io/autoscaler/vertical-pod-autoscaler/pkg/apis/autoscaling.k8s.io/v1" +) + +func NewFakeClient(t *testing.T, initialObjects ...runtime.Object) client.Client { + s := runtime.NewScheme() + for gvk := range scheme.Scheme.AllKnownTypes() { + obj, err := scheme.Scheme.New(gvk) + require.NoError(t, err) + s.AddKnownTypes(gvk.GroupVersion(), obj) + } + require.NoError(t, monitoringv1.AddToScheme(s)) + require.NoError(t, vpav1.AddToScheme(s)) + return fake.NewClientBuilder().WithRuntimeObjects(initialObjects...).WithScheme(s).Build() +}